在刚接触到这个方法时,我只了解到这个是将一个对象添加到列表末尾,具体使用方法如下:
list = [1,2,3]
list.append(4)
# 得到的新的列表就变成了[1,2,3,4]
但是后来在学习的时候遇到这样一个问题,代码如下:
a = [1,2]
b = []
b.append(a)
print(b)
a.append(0)
print(b)
我预想中的执行结果打印出来的内容应该是一样的,但实际上的执行结果却是:
[[1, 2]]
[[1, 2, 0]]
明明两次打印之间并没有对b操作,那么为什么b会发生改变呢,问题肯定是出在了append方法上,对a使用append方法却能同时改变b,难道这就是传说中的隔山打牛w(゚Д゚)w;
通过查阅资料我了解到,原来append方法是浅拷贝,什么是浅拷贝呢?官方一点的解释是,在python中,对象赋值实际上是对象的引用,当创建一个对象,然后把它赋值给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用,这就是浅拷贝。
听不懂没关系,下面慢慢的讲解。
对于长期学习C语言的童鞋来说,进行变量命名及赋值是比较熟悉的:
int a;
a = 1;
int b;
b = 1
在声明变量的时候,int a,int b,这两条语句为a,b两个变量分别赋予了两块不同的内存空间,然后赋值的时候再将相应的值存储到对应的存储空间;
用形象一点的话来描述就是,int a和int b先创建了两个名字叫a,b的盒子,然后赋值就是讲相应的值放到盒子里,这两个是完全不同的。

但是在python中变量的赋值与C语言是截然不同的,举一个简单的栗子:
a = 1
b = 1
print(id(a))
print(id(b))
# id函数用于获取对象的内存地址
运行结果为:
2021455792
2021455792
通过运行结果可以看出来a,b的内存空间是一样的。
在python中,先生成对象,变量再对对象进行引用,在本例中,1就是对象,然后a再对1进行引用,由于常数是不可变类型,所以1的内存空间是一样的,所以a,b引用的是用一块内存空间。
形象一点来解释就是,先生成一个盒子,盒子里边放着1,然后a,b进行对1引用的时候就是把a,b这两个标签贴到了盒子上。

虽然变量名不一样,但是他们引用的对象是相同的。
好了~下面我们回归正题。
append方法进行浅拷贝就相当于python变量赋值一样,在开始的问题中:
a = [1,2]
b = []
b.append(a)
print(b)
a.append(0)
print(b)
b.append(a)就是对a进行了浅拷贝,结果为b=[[1,2]],但b[0]与a引用的对象是相同的,下面通过代码验证一下:
a = [1,2]
b = []
b.append(a)
print(id(a))
print(id(b[0]))
执行结果为:
23768904
23768904
他们的地址是一样的。

然后是a.append(0),列表是可变类型,这一步在原地址的列表的末尾添加0,原地址的内容被改变了但是地址没有变,所以a和b[0]的内容同时被改变了,这就是为什么对a进行append操作b会跟着发生改变。

发生这些的前提是对同一个地址上的内容进行操作,所以影响了指向该地址的所有变量。如果是进行下面的操作,则是另一种结果:
a = [1,2]
b = []
b.append(a)
print(b)
a = [1,2]
a.append(0)
print(b)
执行结果为:
[[1, 2]]
[[1, 2]]
只是多加了一条语句,b就不会随着a的改变而改变,原因就是第二句a=[1,2]重新生成了一个不同的对象(列表是可变类型的),a引用了这个对象,这时候就跟b没有关系了。
