这条线的原因
year_choice = [i for i in range(100) if i > max_year]
在Python 2中工作但在Python 3中工作的是在Python 3中,列表推导创建了一个新的范围,而max_year类属性不在该范围内.在Python 2中,列表推导不会创建新的范围,它在周围代码的上下文中运行.这最初是出于性能原因而做的,但是很多人发现它令人困惑,因此在Python 3中进行了更改,使列表推导符合生成器表达式,以及set和dict comprehension.
AFAIK,在Python 3中没有简单的方法来访问在类的外部上下文中运行的列表解析中的类属性,而不是在方法内部.你不能用Student.max_year引用它,因为那时Student类不存在.
但是,无论如何,那里的列表理解确实没有意义.您可以更紧凑,更高效地创建所需的列表.例如:
class Student(object):
max_year = 18
year_choice = list(range(max_year + 1, 100))
def __init__(self, name):
self.name = name
print (self.year_choice)
Student('Blah')
产量
[19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99]
该代码在Python 2和Python 3上生成相同的输出.
我已将类签名更改为
class Student(object):
这样它就可以在Python 2中创建一个新式的类(在Python 3中,所有类都是新式的).
[i for range in range(max_year)]可以解决这个限制的原因是迭代器是从范围(max_year)创建的,然后作为参数传递给运行列表推导的临时函数.所以它相当于这段代码:
class Student(object):
max_year = 18
def _listcomp(iterator):
result = []
for i in iterator:
result.append(i)
return result
year_choice = _listcomp(iter(range(max_year + 1, 100)))
del _listcomp
def __init__(self, name):
self.name = name
print(self.year_choice)
Student('Blah')
非常感谢Antti Haapala的解释.