您好, 欢迎来到 !    登录 | 注册 | | 设为首页 | 收藏本站

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

5b51 2022/1/14 8:22:18 python 字数 5189 阅读 525 来源 www.jb51.cc/python

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说:

概述

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.

总结

以上是编程之家为你收集整理的使用python3在列表生成器表达式中未定义全局,与python2一起使用,需要哪些更改?全部内容,希望文章能够帮你解决使用python3在列表生成器表达式中未定义全局,与python2一起使用,需要哪些更改?所遇到的程序开发问题。


如果您也喜欢它,动动您的小指点个赞吧

除非注明,文章均由 laddyq.com 整理发布,欢迎转载。

转载请注明:
链接:http://laddyq.com
来源:laddyq.com
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


联系我
置顶