0

0

Python列表底层是什么_CPython源码中List的动态数组原理

P粉602998670

P粉602998670

发布时间:2026-03-19 00:59:17

|

288人浏览过

|

来源于php中文网

原创

Python列表底层是动态数组而非链表,支持O(1)索引访问和len()查询,但头插/头删为O(n);append平均O(1),insert(0,x)始终O(n),大列表头插性能极差,应改用deque。

python列表底层是什么_cpython源码中list的动态数组原理

Python列表底层是动态数组,不是链表

Python的list在CPython中用的是连续内存块实现的动态数组,不是节点指针串联的链表。这意味着list支持O(1)索引访问,但尾部以外的插入/删除是O(n)——因为要移动后续所有元素。

常见错误现象:list.insert(0, x)list.pop(0)在大列表上明显卡顿;有人误以为list适合高频头插,结果性能崩了。

  • 实际存储结构:一个PyListObject结构体,含指向PyObject **数组的指针、当前长度ob_size、已分配容量allocated
  • 扩容策略:当ob_size == allocated时,新容量按(size + 1) * 1.125向上取整(有最小增幅兜底),避免频繁 realloc
  • 缩容时机:仅在ob_size 且<code>allocated > 50时触发,防止“抖动”

为什么append比insert快得多

list.append()只在数组末尾写入,只要容量够就纯O(1);而list.insert(i, x)必须把索引i及之后所有元素整体后移一位,最坏情况(i=0)要搬动全部元素。

使用场景:高频追加用append;需要头插/中间插时,优先考虑collections.deque(双端队列,头插O(1))。

立即学习Python免费学习笔记(深入)”;

  • append平均摊还时间复杂度O(1),因扩容成本被分摊
  • insert(0, x)每次都是O(n),n越大越慢,10万次头插可能比10万次尾插慢百倍
  • CPython源码里list_insert函数会调用memmove搬内存,这是实打实的CPU和缓存开销

len()为什么是O(1)而不是遍历计数

因为list对象自己维护着ob_size字段,len()直接返回它,不扫描数组。

度加创作工具
度加创作工具

百度出品的、人人可用的AIGC创作平台

下载

容易踩的坑:有人写while i 放在循环里,以为<code>len()有开销,其实完全没必要缓存——它就是一次整数读取。

  • len()对应CPython里的list_len函数,一行return Py_SIZE(self)
  • 对比strtuplebytes等不可变序列,也都是O(1),但自定义类若没实现__len__或实现得差,就可能退化
  • 注意:len()不等于list.allocated,后者是总容量,前者是当前有效长度

查看实际内存布局:sys.getsizeof和list.__sizeof__

sys.getsizeof([])返回约56字节(空列表基础开销),每多一个元素,只增加8字节(64位系统下指针大小),但不包括元素自身占用的内存。

真实例子:sys.getsizeof([1, 2, 3]) ≈ 80字节,而[1]*1000[1]*100只多约7200字节(900×8),不是10倍增长——这说明底层确实是紧凑数组,不是每个元素带额外头信息。

  • 想看分配容量?CPython未暴露allocated,但可用list.__sizeof__()粗略估算,或用ctypes读内存(不推荐生产环境)
  • 注意:小整数、短字符串等会被缓存或内联,sys.getsizeof不反映这部分共享内存
  • 真正影响内存的关键是allocated而非len()——比如l = [1]*1000; l.extend([2]*100)后,allocated可能远大于1100

动态数组的“动态”二字藏在扩容缩容逻辑里,不是简单 realloc;而“数组”意味着连续性——这点一旦被忽略,比如当成链表来设计算法,性能问题就很难靠后期优化补救。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

781

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

696

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

784

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

890

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1564

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

598

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

591

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

759

2023.08.11

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

7

2026.03.18

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 5.1万人学习

SciPy 教程
SciPy 教程

共10课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号