首先
>>> a=3
>>> b=a
b=a会使python创建变量b。变量a正在使用,并且a这里没有被赋值,所以a被替换成其引用的对象3,从而b也成为这一对象的引用。a和b指向了相同的内存空间。这在Python中叫做共享引用——多个变量名引用了同一个对象。
下一步
>>> a=3
>>> b=a
>>> a='spam'
第三条语句创建了新的对象,并把a指向它。a的值发生了改变,b没有改变。同样如果只将b=spam,a也不会受到影响。
在Python中,变量总是一个指向对象的指针,而不是可改变的内在区域的标签:给一个变量赋一个新值,并不是替换了原始的对象,而是让这个变量去引用完全不同的一个对象。实际的效果就是对一个变量赋值,仅仅会影响那个被赋值的变量。
共享引用和在原处修改
有一些对象和操作确实会在原处改变对象。例如,在一个列表中对一个偏移进行赋值确实会改变这个列表对象,而不是生成一个新的列表对象。对于支持这种在原处修改的对象,一个变量的修改会影响到其他的变量。
>>> L1=[2,3,4]
>>> L2=L1
>>> L1=24
>>> L1
24
>>> L2
[2, 3, 4]
以上代码和上一则代码段同理,新建了一个对象。而下面则不同:
>>> L1=[2,3,4]
>>> L2=L1
>>> L1[0]=24
>>> L1
[24, 3, 4]
>>> L2
[24, 3, 4]
在这里,没有改变L1,却改变了L1所引用对象的一个元素。这类修改会覆盖列表对象中的某部分。因为这个列表对象与其他对象共享的(被其他对象引用)。在这个例子中,也会对L2产生影响,因为它与L1都引用了相同的对象。
想避免这种情况可以使用拷贝。
分片:
>>> L1=[2,3,4]
>>> L2=L1[:]
>>> L1[0]=24
>>> L1
[24, 3, 4]
>>> L2
[2, 3, 4]
这里对L1的修改不会影响L2,因为L2引用的是L1所引用的一个拷贝,也就是说,两个变量指向了不同的内存区域。注意这种分片技术不会应用在其他的可变的核心类型(字典和集合,因为它们不是序列)上,复制一个字典或集合应该使用x.copy()方法调用。