首先,需要明白的是python是一种解释性语言标准,他可以由c、java或者其他语言来实现。在python的标准中,变量是链接式的。举个例子来说,内存中的对象和变量就像是给内存对象贴变量名标签而不是给变量盒子装入东西(数值信息)。或者说是给对象分配变量名,而不是给变量赋值。因为python对象在执行赋值之前就已经创立存在了的。
id()返回对象的标识,cpython中id()返回的是唯一的内存地址,变量名则是一个便于理解的别名。
is需要值和内存地址都相等,==只需要值相等。检查一下id()就可以了。
is比==更快,不能重载,直接比较的是id。
在基础阶段,很多教程都会提到,元祖是不可变的数据类型,所以没有插入和删除的操作,只能一次定义。但深究这个问题起来,其实元祖只是在引用上不能变化,但引用的对象是可变的,就导致元祖的值会跟着列表发生变化。浅复制时,多个变量引用同一列表地址,那么改变一个其他也会变。深复制是在内存中重新开辟一个区域,将原来的对象包括其中为对象引用的内容,一同复制到新的区域,并建立新的引用。引用变量的可变性看的是最小对象单位的类型,比如不可变元组中有可变列表对象,列表就是最小的单位,要看它的可变性。列表是可变的,这种特性使得在指向同一个列表对象的情况下,列表实际的长度和值是可以改变的。对其他的类型对象一般不可变,也就不会发生上述变化。
最大的威胁在于,将列表作为参数传给类方法的时候,类变量和参数绑定了同一个对象,执行完返回后原来的传入参数的对象发生了变化。除了刻意要去改传入的列表,否则应该在这之前深复制一个副本,不影响原来的变量的使用。
a = [1,2,3]
b=a
b is a
>>True
c = list(a)
c is a
>>False
d = a[:]
d is a
>>False
类似上面使用b=a这样的办法创建的列表,其实只是新建了一个引用,并没有新建一个对象。如果改变a、b其中一个,另一个会随着改变。
copy模块可以完成浅复制和深复制,分别对应方法copy()和deepcopy()。其中,deepcopy()可以完成对包含循环引用的对象的拷贝。
具体可以用下面的a分别做两种拷贝,会发现其中深拷贝得到的a[1]的内容和另外两者是不一一样的。
a = [1,[2,3],(4,5)]
使用两种拷贝分别复制后查看结果。
原文地址:https://www.cnblogs.com/xiao3c/p/9592139.html