
本文旨在解释Python中列表赋值的特性,重点在于区分修改现有列表与创建新列表的区别。通过具体示例,阐明当一个列表赋值给另一个列表时,它们共享同一对象,因此修改其中一个会影响另一个。同时,解释了重新赋值列表变量会创建新对象,从而避免相互影响。掌握这些概念对于编写可预测和避免意外副作用的Python代码至关重要。
在Python中,列表的赋值行为与其他一些编程语言有所不同,理解这种差异对于编写高效且避免bug的代码至关重要。核心在于Python的变量实际上是对对象的引用,而非对象本身。
列表赋值与引用
当使用second = first这样的语句时,你并没有创建first列表的副本。实际上,second只是成为了first所指向的同一个列表对象的另一个引用。这意味着,如果通过second修改列表,first也会受到影响,因为它们指向的是内存中的同一个列表。
立即学习“Python免费学习笔记(深入)”;
first = [1, 2, 3, 4, 5] second = first second.append(6) print(first) # 输出: [1, 2, 3, 4, 5, 6] print(second) # 输出: [1, 2, 3, 4, 5, 6]
在这个例子中,second.append(6)修改了列表,由于first和second引用的是同一个列表,所以first也反映了这一变化。
列表重新赋值
与修改现有列表不同,如果将一个新的列表赋值给first,则会创建一个新的列表对象,并将first指向这个新对象。而second仍然指向原来的列表对象,因此first的变化不会影响second。
first = [1, 2, 3, 4, 5] second = first first = [2, 4, 6, 8] print(first) # 输出: [2, 4, 6, 8] print(second) # 输出: [1, 2, 3, 4, 5]
在这个例子中,first = [2, 4, 6, 8]创建了一个新的列表对象,并将其赋值给first。second仍然指向原来的列表[1, 2, 3, 4, 5]。
如何创建列表的副本
如果希望创建一个列表的副本,而不是创建一个新的引用,可以使用以下方法:
- 切片: second = first[:]
- list() 构造函数: second = list(first)
- copy() 方法: second = first.copy() (Python 3.3+)
- deepcopy() 方法: 如果列表包含可变对象(例如,嵌套列表),则需要使用copy.deepcopy()来创建深拷贝,以确保所有嵌套对象也被复制。
这些方法都会创建一个新的列表对象,其内容与原始列表相同,但它们是独立的,修改其中一个不会影响另一个。
import copy first = [1, 2, [3, 4]] second = first[:] # 或者 first.copy() third = copy.deepcopy(first) second[0] = 10 second[2][0] = 30 third[0] = 20 third[2][0] = 40 print(first) # 输出: [1, 2, [30, 4]] print(second) # 输出: [10, 2, [30, 4]] print(third) # 输出: [20, 2, [40, 4]]
在这个例子中,使用[:]切片创建了second,使用deepcopy()创建了third。修改second和third不会影响first,但修改second中的嵌套列表会影响first,因为浅拷贝只复制了顶层列表,而嵌套列表仍然是共享的。deepcopy() 则完全复制了所有层级的对象。
总结与注意事项
- 理解Python中变量是对对象的引用,而非对象本身。
- second = first 创建的是一个引用,而非副本。
- 修改通过引用指向的列表会影响所有引用该列表的变量。
- 使用切片、list()构造函数或copy()方法创建列表的浅拷贝。
- 使用copy.deepcopy()创建列表的深拷贝,尤其是在处理包含可变对象的嵌套列表时。
掌握这些概念可以帮助你避免在Python编程中出现意外的副作用,并编写出更清晰、可维护的代码。










