0

0

PHP 数组底层实现原理面试题

冷炫風刃

冷炫風刃

发布时间:2026-03-10 13:16:03

|

150人浏览过

|

来源于php中文网

原创

php数组既是“数组”又是“字典”因其底层hashtable融合双向链表(保插入顺序)与哈希桶数组(o(1)查找),统一存储zval并支持整数/字符串键,结合写时复制与惰性缩容机制。

php 数组底层实现原理面试题

PHP 数组底层是用 哈希表(HashTable) 实现的,它同时支持整数下标和字符串键名,本质是一个“有序的、可重复索引的、支持混合键类型的映射结构”。这不是传统意义的 C 数组,也不是纯哈希容器,而是 PHP 自研的高效混合结构。

为什么 PHP 数组既是“数组”又是“字典”?

因为底层 HashTable 内部维护了一个 双向链表 + 哈希桶数组 的组合结构:

  • 哈希桶数组(buckets)负责 O(1) 平均查找:根据 key 的 hash 值定位槽位,解决键名查找问题;
  • 双向链表(arData)保持插入顺序:每个元素按写入顺序串在链表中,所以 foreach 遍历一定按插入顺序;
  • 所有元素(zval)统一存放在连续内存块中(arData),通过索引快速访问,支撑整数下标操作(如 $arr[0]);
  • 整数 key 会尝试转为“packed array”优化路径(如全为连续非负整数时,跳过哈希计算,直接按偏移访问)。

数组扩容与 rehash 是怎么发生的?

HashTable 有容量(nTableSize)和实际元素数(nNumOfElements)。当插入导致负载因子(nNumOfElements / nTableSize)超过阈值(通常是 0.75),就会触发扩容:

蛙蛙写作——超级AI智能写作助手
蛙蛙写作——超级AI智能写作助手

蛙蛙写作辅助AI写文,帮助获取创意灵感,提供拆书、小说转剧本、视频生成等功能,是一款功能全面的AI智能写作工具。

下载
  • 分配新桶数组,大小翻倍(如从 8→16),且一定是 2 的幂(便于位运算取模);
  • 对所有已有元素重新计算 hash,再插入新桶中(即 rehash);
  • 注意:rehash 不改变元素在双向链表中的顺序,只更新哈希桶指针;
  • 删除元素不会缩容,但 PHP 8.0+ 引入了惰性缩容机制(如连续多次删除后可能触发)。

zval 和数组如何协同工作?

PHP 7+ 中,数组内部存储的是 zval 的值拷贝或引用(非指针),而 zval 本身包含类型、值、引用计数(refcount)和是否为垃圾(gc_info)等字段:

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

  • 赋值 $b = $a 时,不立即复制整个 HashTable,而是共享底层数据,仅增加 refcount;
  • 当某一方发生写操作(如 $b[] = 1),才触发“写时复制(Copy On Write)”,分离出独立副本;
  • 空数组([])初始化时,HashTable 结构已存在,但 arData 为空,桶数组最小为 8;
  • unset 某个 key 后,该位置在链表中被标记为“空洞(empty slot)”,仍占链表节点,但桶中指针置空,后续插入可能复用。

常见误区提醒

面试中容易答偏的点:

  • ❌ “PHP 数组是用红黑树实现的” —— 错,是哈希表 + 双向链表;
  • ❌ “键名顺序由哈希值决定” —— 错,遍历顺序由链表决定,和哈希无关;
  • ❌ “整数键和字符串键存储在不同区域” —— 错,统一存于同一套 arData + HashTable 中,只是 hash 计算方式不同;
  • ✅ 关键结论:PHP 数组 = 有序性(链表) + 快速查找(哈希) + 内存友好(连续 zval 块) + 写时复制(COW)。

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

265

2025.12.04

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1566

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1208

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1184

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

192

2025.07.29

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 13.2万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.3万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 1.0万人学习

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

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