我遇到了一些在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
import Tkinter, MySQLdb db=MySQLdb.connect(#blahblah) ; cur=db.cursor() #cur is defined! from utilities_module import *
但这是循环导入或类似的操作,最重要的是,它也崩溃了。所以我的问题是:
我该如何在主模块中定义的“ cur”对象对导入其中的那些辅助功能可见?
如果你将解决方案发布在其他地方,则感谢你的宝贵时间和最深切的歉意。我只是自己找不到答案,而且我的书中没有其他花招。
Python中的全局变量是模块的全局变量,而不是所有模块的全局变量。(许多人对此感到困惑,因为在C语言中,所有实现文件中的全局变量都是相同的,除非你明确指定了它static。)
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。
from shared_stuff import a
shared_stuff.a
或者,在极少数情况下,你确实确实需要它在任何地方都具有真正的全局性(例如内置),将其添加到内置模块中。确切的细节在Python 2.x和3.x之间有所不同。在3.x中,它的工作方式如下:
import builtins import module1 builtins.a = 3 module1.f()