问题:导入模块中全局变量的可见性
我在使用Python脚本导入模块时遇到了一些麻烦。我将尽力描述错误,为什么会遇到错误以及为什么要使用这种特殊方法来解决我的问题(我将在稍后描述):
假设我有一个模块,其中定义了一些实用程序函数/类,这些函数/类引用在此辅助模块将导入到的命名空间中定义的实体(让“ a”是这样的实体):
模块1:
def f():
print a
然后,我有了主程序,其中定义了“ a”,我要将这些实用程序导入其中:
import module1
a=3
module1.f()
执行该程序将触发以下错误:
Traceback (most recent call last):
File "Z:\Python\main.py", line 10, in <module>
module1.f()
File "Z:\Python\module1.py", line 3, in f
print a
NameError: global name 'a' is not defined
过去(两天前,d’uh)曾提出过类似的问题,并提出了几种解决方案,但是我真的不认为这些符合我的要求。这是我的特定情况:
我正在尝试制作一个Python程序,该程序连接到MySQL数据库服务器并使用GUI显示/修改数据。为了简洁起见,我在一个单独的文件中定义了一堆与MySQL相关的辅助/实用程序功能。但是它们都有一个公共变量,该变量是我最初在实用程序模块中定义的,并且是MySQLdb模块中的游标对象。后来我意识到,游标对象(用于与db服务器通信的对象)应该在主模块中定义,以便主模块和导入到其中的所有对象都可以访问该对象。
最终结果将是这样的:
utilities_module.py:
def utility_1(args):
code which references a variable named "cur"
def utility_n(args):
etcetera
而我的主要模块:
program.py:
import MySQLdb, Tkinter
db=MySQLdb.connect(#blahblah) ; cur=db.cursor() #cur is defined!
from utilities_module import *
然后,一旦我尝试调用任何实用程序函数,就会触发上述“未定义全局名称”错误。
一个特别的建议是在实用程序文件中有一个“ from program import cur”语句,例如:
utilities_module.py:
from program import cur
#rest of function definitions
program.py:
import Tkinter, MySQLdb
db=MySQLdb.connect(#blahblah) ; cur=db.cursor() #cur is defined!
from utilities_module import *
但这是循环导入或类似的操作,最重要的是,它也崩溃了。所以我的问题是:
我该如何在主模块中定义的“ cur”对象对导入到其中的辅助功能可见?
如果您将解决方案发布在其他位置,则感谢您的宝贵时间和最深切的歉意。我只是自己找不到答案,而且我的书中没有其他花招。
回答 0
Python中的全局变量是模块的全局变量,而不是所有模块的全局变量。(许多人对此感到困惑,因为在C语言中,除非您明确创建全局变量,否则所有实现文件中的全局变量都是相同的static
。)
有多种解决方法,具体取决于您的实际用例。
在走这条路之前,请问自己这是否真的需要是全球性的。也许您真的想要一个带有f
实例方法的类,而不仅仅是一个自由函数?然后,您可以执行以下操作:
import module1
thingy1 = module1.Thingy(a=3)
thingy1.f()
如果您确实确实想要一个全局变量,但是它只是供您使用module1
,请在该模块中进行设置。
import module1
module1.a=3
module1.f()
另一方面,如果a
由许多模块共享,则将其放置在其他位置,并让每个人都将其导入:
import shared_stuff
import module1
shared_stuff.a = 3
module1.f()
…并且,在module1.py中:
import shared_stuff
def f():
print shared_stuff.a
from
除非变量打算是一个常量,否则不要使用导入。from shared_stuff import a
会创建一个新a
变量,初始化为shared_stuff.a
导入时所引用的变量,并且该新a
变量将不受分配的影响shared_stuff.a
。
或者,在极少数情况下,您确实确实需要它在任何地方都具有真正的全局性(例如内置),将其添加到内置模块中。确切的细节在Python 2.x和3.x之间有所不同。在3.x中,它的工作方式如下:
import builtins
import module1
builtins.a = 3
module1.f()
回答 1
解决方法是,您可以考虑像这样在外层设置环境变量。
main.py:
import os
os.environ['MYVAL'] = str(myintvariable)
mymodule.py:
import os
myval = None
if 'MYVAL' in os.environ:
myval = os.environ['MYVAL']
作为额外的预防措施,请在模块内部未定义MYVAL的情况下进行处理。
回答 2
函数使用其定义模块的全局变量。a = 3
例如,应该设置而不是set module1.a = 3
。因此,如果要cur
用作全局输入utilities_module
,请设置utilities_module.cur
。
更好的解决方案:不要使用全局变量。将所需的变量传递到需要它的函数中,或者创建一个类将所有数据捆绑在一起,并在初始化实例时传递它。
回答 3
这篇文章只是我遇到的Python行为的观察。如果您做的事情与我在下面做的相同,则上面阅读的建议可能对您不起作用。
即,我有一个包含全局/共享变量的模块(如上所述):
#sharedstuff.py
globaltimes_randomnode=[]
globalist_randomnode=[]
然后,我有一个主要模块,用于导入共享内容:
import sharedstuff as shared
以及实际填充这些数组的其他一些模块。这些由主模块调用。当退出这些其他模块时,我可以清楚地看到已填充了阵列。但是,当在主模块中重新读取它们时,它们为空。这对我来说很奇怪(嗯,我是Python的新手)。但是,当我将主模块中的sharedstuff.py导入方式更改为:
from globals import *
它有效(填充了数组)。
只是在说’
回答 4
解决此特定问题的最简单方法是在模块内添加另一个功能,该功能会将光标存储在模块的全局变量中。然后所有其他功能也可以使用它。
模块1:
cursor = None
def setCursor(cur):
global cursor
cursor = cur
def method(some, args):
global cursor
do_stuff(cursor, some, args)
主程序:
import module1
cursor = get_a_cursor()
module1.setCursor(cursor)
module1.method()
回答 5
由于全局变量是特定于模块的,因此可以将以下函数添加到所有导入的模块中,然后将其用于:
- 将单数变量(以字典格式)添加为这些变量的全局变量
- 将您的主模块全局变量传递给它。
addglobals = lambda x:globals()。update(x)
然后,您需要传递当前的全局变量是:
导入模块
module.addglobals(globals())
回答 6
由于我在上面的答案中没有看到它,因此我想我将添加一个简单的解决方法,global_dict
即向需要调用模块全局变量的函数添加一个参数,然后在调用时将dict传递给该函数。例如:
# external_module
def imported_function(global_dict=None):
print(global_dict["a"])
# calling_module
a = 12
from external_module import imported_function
imported_function(global_dict=globals())
>>> 12
回答 7
这样做的OOP方法是使模块成为类,而不是一组未绑定的方法。然后,您可以使用__init__
或setter方法来设置来自调用方的变量,以用于模块方法中。