Python列表底层是动态指针数组,含引用计数与过量分配机制;append()平摊O(1)因扩容按new_allocated = (size >> 3) + (size
Python 列表不是“可变数组”的简单翻译,它的底层是动态指针数组 + 引用计数 + 过量分配(over-allocation)机制。直接操作
list时,你其实一直在和这套内存管理策略打交道。为什么
append()平摊时间复杂度是 O(1),但单次可能触发 O(n) 重分配?CPython 的
list在扩容时,并非每次只加 1 个槽位,而是按公式new_allocated = (size >> 3) + (size 增长(见 listobject.c)。这意味着:
- 小列表(如长度
- 大列表(如长度 1000)再
append(),可能新增约 125 个空位- 真正耗时的是
memcpy整块复制旧数据到新地址——这步不可省略,且发生在扩容瞬间- 所以连续调用 1000 次
append(),实际只重分配约 10–15 次,平摊下来接近常数
del lst[i]和lst.pop()的性能差异远不止“删尾 vs 删中”删除末尾元素(
pop())只需将ob_size减 1;而删除中间或开头元素(del lst[i])必须把i+1到末尾的所有指针向前挪一位——这是纯 C 级别的内存移动:import timeit lst = list(range(100000)) timeit.timeit(lambda: lst.pop(), number=100000) # ≈ 0.012s timeit.timeit(lambda: del lst[0], number=100000) # SyntaxError —— 正确写法是: timeit.timeit(lambda: lst.__delitem__(0), number=100000) # ≈ 2.8s(慢 200 倍以上)更隐蔽的坑:
lst.remove(x)先遍历找索引,再执行__delitem__,等价于 O(n) 查 + O(n) 移。立即学习“Python免费学习笔记(深入)”;
用
list.extend()替代循环append()不只是为了“写得短”假设你要合并两个列表:
for x in other: target.append(x)→ 每次append都可能触发检查、扩容、复制target.extend(other)→ C 层直接预估总长度,一次分配到位,再批量 memcpy- 若
other是生成器(如range(10**6)),extend仍能高效处理;而循环append会因反复扩容严重拖慢实测:向空列表添加 100 万个整数,
extend(range(10**6))比循环append快 3–5 倍。别依赖
id(lst)不变来判断“列表没重建”,它掩盖了真实风险看似安全的操作,比如:
lst = [1, 2, 3] original_id = id(lst) lst += [4, 5] # 原地修改,id 不变 lst *= 2 # 原地修改,id 不变 lst = lst + [6] # 创建新对象!id 已变问题在于:
+=和*=对list是就地操作(调用list_inplace_concat),但+和*总是新建对象。如果你在函数外持有原列表引用,又误用+赋值,就可能引发静默的引用失效。真正需要关注的不是
id,而是是否触发了底层realloc或指针复制——这些对上层透明,但影响缓存局部性和 GC 压力。
0
0
相关文章
Python装饰器系统学习路线第246讲_核心原理与实战案例详解【指导】
可以同时发布开发版和稳定版:版本管理最佳实践
Python日志监控系统学习路线第519讲_核心原理与实战案例详解【指导】
Python健壮代码设计原则_异常与边界处理策略【教程】
Python多进程系统学习路线第17讲_核心原理与实战案例详解【技巧】
本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门AI工具
相关专题
php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。
709
2023.06.15
python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。
736
2023.07.25
Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。
616
2023.07.31
Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。
547
2023.08.04
eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。
573
2023.08.04
scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。
695
2023.08.11
热门下载
相关下载
精品课程
最新文章





