Python 不可变对象真的不可变吗?

在日常的工作和学习中,经常会遇到“不可变对象 ” 相关的问题,但是随着接触Python这门语言的时间越来越多,遇到的坑越来越奇怪。我不禁产生了一个疑问:不可变对象真的不可变吗?

我们知道元组就是”不可变对象”,当你想尝试给一个元组赋值的时候,它会报错:

>>> my_tuple = ("python", "dict", "is", "good")
>>> my_tuple[3] = "excellent" 
Traceback (most recent call last):   
File "<stdin>", line 1, in <module> 
TypeError: 'tuple' object does not support item assignment 

请注意报错的语言: “does not support item assignment “, 元组对象不支持赋值操作。

例1

让我们来看看下面这样的操作:

>>> int_tuple = ([6,5,30], [2,3,65], [111])
>>> int_tuple[2] += [22,33]
Traceback (most recent call last):  File "<stdin>", line 1, in <module>
TypeError: 'tuple' object does not support item assignment
>>> int_tuple([6, 5, 30], [2, 3, 65], [111, 22, 33]) 

尽管报了一个错,但是它变量的值已经被改变了。

原因

其实,这里主要是因为 += 操作对于“不可变对象”会产生新的变量并绑定到原有变量上。但是又由于+=中的”=”号,执行了一次对元组对象的赋值操作,这是不允许的,因此报了错,但新的对象已经被绑定到了原有变量中,因此我们可以看到变量的值发生了改变。

例2

还有一种情况是这样的:

>>> int_tuple = ([6,5,30], [2,3,65], [111])
>>> int_tuple[2].append(123123)
>>> int_tuple
([6, 5, 30], [2, 3, 65], [111, 123123]) 

这里完全没有报错,为什么呢?因为append并不涉及到赋值操作,元组只是不允许赋值,并没有不允许append和extend啊,所以称元组为“不可变对象” 实在是太不严谨了!

深一点

其实如果你知道元组中存放的是元素所对应的地址(ID),就好理解多了,append和extend仅仅是修改了列表的元素,而列表本身的ID并没有发生变化,只有当赋值操作执行的时候,ID才会发生变化,而这种情况是元组不允许发生的。

# append 不会发生地址变化
>>> int_tuple = ([1], [2], [3])
>>> id(int_tuple[1])
2729947990600
>>> int_tuple[1].append(22)
>>> id(int_tuple[1])
2729947990600 

因此,要避免踩上这些坑,忘掉元组是“不可变对象”这样不严谨的表述形式(当然从地址的角度来讲它是对的,元组内部所有元素的ID都不可变),请这样记:元组是不可赋值对象

我们的文章到此就结束啦,如果你希望我们今天的Python 教程,请持续关注我们,如果对你有帮助,麻烦在下面点一个赞/在看哦有任何问题都可以在下方留言区留言,我们都会耐心解答的!


​Python实用宝典 (pythondict.com)
不只是一个宝典
欢迎关注公众号:Python实用宝典

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注