当我将一个新元素附加到TUPLE时,为什么它可以工作?

时间:2017-01-17 23:23:26

标签: python list python-3.x tuples mutable

由于元组是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?")

2 个答案:

答案 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

轰!