class Some(object): tokens = [ ... list of strings ... ] untokenized = [tokens.index(a) for a in [... some other list of strings ...]] ... etc ... some = Some()
这可以很好的Python2.7.不过python3说:
Traceback (most recent call last): File "./test.py",line 17,in <module> class Some(object): File "./test.py",line 42,in Some untokenized = [tokens.index(a) for a in [... some other list of strings ...]] File "./test.py",in <listcomp> untokenized = [tokens.index(a) for a in [... some other list of strings ...]] NameError: global name 'tokens' is not defined
虽然我可以解决这个问题,但我真的很想知道Python2和Python3之间有什么区别.我读过python 2-> 3更改文档,但是我无法识别与我的问题有关的任何描述. 2to3工具也不会在我的代码中抱怨任何东西.
顺便说一句,虽然我现在不记得了,但是我也有类似于python2的东西(我甚至没有用3尝试过),我认为这应该工作(在一个类内):
def some_method(self): return {a: eval("self." + a) for a in dir(self) if not a.startswith("_")}
但是它引起python2说:NameError:name’self’没有定义
我还没有尝试过这个python3,但例如这个工作:
[eval("self." + a) for a in dir(self) if not a.startswith("_")]
如果我将上一个例子的相关部分更改为这个例子(好的,这个例子本身有点笨,但至少显示了我的问题).现在我很好奇,为什么自己似乎没有被定义为这个第一个例子,但它是第二个?似乎与dicts,我有类似的问题,我的原始问题是关于,但与列表生成器表达式它的工作,但不是在python3.嗯…
之后我的python2 – > 3问题我提到这一点,因为所有这些似乎都是关于某些东西根据python解释器定义的问题(也许我的问题的第二部分是不相关的).我现在觉得很困惑请告诉我关于我的错误(因为我确信我错过了一些事情).
解决方法
class C: a = 2 b = a + 2 # b = 4
但在类的体内引入的范围无法访问其命名空间:
class C: a = 2 def foo(self): return a # NameError: name 'a' is not defined,use return self.__class__.a
Python 2和Python 3之间的区别在于,在Python 2列表推导中不会引入新的范围:
[a for a in range(3)] print a # prints 2
而在Python 3中,他们做:
[a for a in range(3)] print(a) # NameError: name 'a' is not defined
这在Python 3中有所改变,原因有两个,其中包括使列表推导的行为与generator-expressions(genexps)相同; (a for a in range(3))在Python 2和Python 3中都有自己的范围.
因此,在一个类的体内,Python 2 genexp或Python 3的listcomp或genexp引入了一个新的范围,因此无法访问类定义的本地命名空间.
给genexp / listcomp访问类定义命名空间中的名称的方式是引入一个新的范围,使用一个函数或一个lambda:
class C: a = 2 b = (lambda a=a: [a + i for i in range(3)])()
eval问题
您的eval示例的问题是eval默认情况下会在本地范围内评估其参数;因为Python 2列表推导具有上述共享封闭范围的行为,eval可以访问方法范围,但是genexp或Python 3 listcomp本地作用域只具有封闭范围所需的编译器所需的任何内容(因为genexp / listcomp范围是封闭):
def bar(x): return list(eval('x') + x for i in range(3)) bar(5) # returns [10,10,10] def baz(x): return list(eval('x') for i in range(3)) baz(5) # NameError: name 'x' is not defined
正如Martijn所说,而不是eval,你应该使用getattr.