使用python3在列表生成器表达式中未定义全局,与python2一起使用,需要哪些更改?

前端之家收集整理的这篇文章主要介绍了使用python3在列表生成器表达式中未定义全局,与python2一起使用,需要哪些更改?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
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解释器定义的问题(也许我的问题的第二部分是不相关的).我现在觉得很困惑请告诉我关于我的错误(因为我确信我错过了一些事情).

解决方法

正如Wooble所说,问题是类不具有词汇范围(实际上在 Python 2Python 3).相反,它们具有不构成范围的本地命名空间.这意味着类定义中的表达式可以访问命名空间的内容
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.

猜你在找的Python相关文章