python常见误解包括:可变对象作默认参数会复用同一对象;is比较身份而非值,应慎用;遍历中删列表元素易漏项;赋值仅绑定名字不复制对象,需用copy或deepcopy避免意外修改。

Python 看似简单,但不少初学者甚至有经验的开发者,常因直觉或类比其他语言而产生误解。这些误解往往导致难以察觉的 bug、性能问题或设计偏差。关键在于:Python 的行为由其对象模型、内存管理机制和运行时特性决定,而非表面语法。
可变对象作为函数默认参数是危险的
很多人以为 def func(items=[]) 中的空列表每次调用都会“重新创建”,实际它只在函数定义时创建一次,并在后续所有未传参的调用中复用同一个列表对象。
例如:
>>> def append_to(a, lst=[]):lst.append(a)
return lst
>>> append_to(1)
[1]
>>> append_to(2)
[1, 2] ← 意外累积
正确做法是用 None 作占位符,在函数体内显式初始化:
立即学习“Python免费学习笔记(深入)”;
- 写成
def append_to(a, lst=None): - 开头加判断:
if lst is None: lst = [] - 这样每次调用都获得独立的新列表
is 和 == 完全不是一回事
== 比较的是值是否相等(调用 __eq__),而 is 比较的是两个变量是否指向**同一个对象**(即内存地址是否相同)。
常见误用:用 if x is True: 或 if x is None: 是合理的(因为单例),但 if x is [1,2,3]: 几乎总错——即使内容一样,也是不同列表对象。
- 判断布尔值直接用
if x:更 Pythonic - 判断是否为 None,用
is None安全且推荐 - 判断内容相等,一律用
==;除非你明确需要身份比较
循环中修改列表,容易漏元素或索引越界
一边遍历列表一边用 remove() 或 del 删除元素,会改变剩余元素的位置,导致下一轮迭代跳过紧邻的下一个元素。
例如:
>>> nums = [1, 2, 3, 4, 5]>>> for n in nums:
if n % 2 == 0:
nums.remove(n)
>>> nums
[1, 3, 5, 4] ← 4 没被检查(因为原索引 3 的元素被删后,4 移到了索引 3,但循环已走到索引 4)
安全做法有三种:
- 反向遍历:
for i in range(len(nums)-1, -1, -1):再按索引删 - 构建新列表:
nums = [n for n in nums if n % 2 != 0] - 用
while+ 显式索引控制,删完不自增索引
赋值从不复制对象,只是绑定名字
a = b 不是“把 b 的值拷贝给 a”,而是让名字 a 指向 b 当前所指的对象。对可变对象来说,这意味修改 a 可能影响 b。
例如:
>>> b = [1, 2, 3]>> a = b
>> a.append(4)
>> b
[1, 2, 3, 4] ← b 也被改了
如需真正独立副本:
- 浅拷贝:
a = b.copy()、a = b[:]、a = list(b) - 深拷贝(含嵌套):
import copy; a = copy.deepcopy(b) - 注意:不可变对象(如 str、int、tuple)无需拷贝,赋值即安全










