由于元组是Python中不可变的数据类型,而元组理解不是一个问题,那么为什么使用圆括号而不是方括号的List理解能够正常工作并生成常规元组?
我认为使用圆括号来定义元组而不是列表(我知道我没有错)。
根据我的理解,我将值附加到已经定义的元组,并且我能够更新元组,并且不应该发生(在Python中),因为元组是不可变的。
我没有收到任何错误,有人请告诉我有一个合理的解释。
这是我的代码:
ignore = (r"^\@define\s")
x = tuple()
(x.append(True if re.search(regex, line) else False) for regex in ignore)
if True in x:
print("How is this possible?")
答案 0 :(得分:9)
由于您使用的是生成器表达式,因此不会引发错误。
>>> x = tuple()
>>> x
()
>>> (x.append(i) for i in range(10))
<generator object <genexpr> at 0x110bede60>
生成器被懒惰地评估,并且因为你甚至没有在变量中捕获生成器,所以它只是被垃圾收集。但是,看看当我做使用生成器时会发生什么:
>>> g = (x.append(i) for i in range(10))
>>> next(g)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 1, in <genexpr>
AttributeError: 'tuple' object has no attribute 'append'
生成器是一种编写迭代器的快速,酷炫的方法。 生成器表达式本质上是“生成器的列表推导”,但您可以使用yield
编写生成器:
>>> def my_generator():
... yield 1
... yield 3
...
>>> g = my_generator()
>>> next(g)
1
>>> next(g)
3
>>> next(g)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration
>>> list(my_generator())
[1, 3]
>>>
现在,关于生成器表达式的一个很酷的事情是你可以将它们与tuple
构造函数结合起来,这构造函数可以使用任何迭代,并且你有一个穷人的元组理解! *注意,如果只有一个参数,则可以将括号放在函数参数中:
>>> tuple(x for x in range(20))
(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19)
>>>
顺便说一句,你应该从不运行这样的代码:
>>> x = []
>>> [x.append(i) for i in range(10)]
[None, None, None, None, None, None, None, None, None, None]
>>> x
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
首先,它是浪费的,因为它创建了一个完全没用的None
列表。其次,它将功能结构,列表理解与状态变化(.append
)混合,这是一种不好的形式。 Python程序员希望列表理解不要这样做。 只需使用for循环执行命令。
答案 1 :(得分:4)
让我们展示一个更简单的例子,让您在REPL中的家中跟随:
>>> def error(): raise Exception
...
>>> (error() for x in range(5))
<generator object <genexpr> at 0x7fb1a371f2d0>
(error() for x in range(5)
是一个genexp - 一个生成器表达式。当它被评估时,它会产生一个生成器对象,它可以根据需要懒洋洋地生成项目 - 但是现在,我们还没有要求它生成任何东西!因此,其中包含的代码都没有运行。
相比之下,让我们看看当我们尝试将生成器的输出扩展为元组时会发生什么:
>>> tuple(_)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 1, in <genexpr>
File "<stdin>", line 1, in error
Exception
轰!