PHP 数组相关面试题系统梳理与总结

6次阅读

php数组实为有序映射,底层基于hashtable,支持整型/字符串key(含隐式转换规则);操作需注意isset与array_key_exists差异、empty判空陷阱、foreach引用残留、array_merge对NULL报错等;算法常用array_multisort、array_intersect系列、array_reduce扁平化;性能上宜缓存count、用range+array_flip建映射、避免循环中频繁array_merge。

PHP 数组相关面试题系统梳理与总结

PHP 数组底层结构与类型辨析

PHP 中的“数组”实际是有序映射(ordered map),既可当索引数组用,也能作关联数组、甚至模拟、队列、哈希表。它底层基于 HashTable 实现,每个元素由 key → value 映射构成,key 可为整型或字符串(其他类型会自动转换:Float 截断、null 变 ”、bool true→1、false→0)。注意:数字字符串 key(如 “123”)会被当作整型处理,导致与纯数字 key 冲突;而 “0123” 这类带前导零的字符串则保留为字符串 key。

常见数组操作陷阱与正确写法

面试常考边界行为和隐式转换:

  • isset($arr[10]) 与 array_key_exists(10, $arr):前者对 null 值返回 false,后者只判断 key 是否存在,更准确;
  • empty($arr) 判空不严谨:空数组返回 true,但含一个值为 0 / “0” / false / null 的数组也会被误判为空,应优先用 count($arr) === 0
  • foreach 遍历时修改数组:PHP 7+ 中 foreach 基于数组副本迭代,内部修改不影响当前循环;但若用 & 引用赋值(foreach ($arr as &$v)),需在循环后 unset($v),否则末尾引用会滞留并污染后续变量;
  • array_merge([], null) 会报 Warning,而 array_replace([], null) 则静默忽略非数组参数。

高频算法题核心解法模式

多数数组算法题围绕去重、查找、合并、变换展开,关键在于选对内置函数组合:

  • 二维数组按字段排序:用 array_multisort(array_column($arr, 'score'), SORT_DESC, $arr),避免手写 usort + 匿名函数;
  • 获取多维数组中所有指定键的值(如所有 ‘id’)递归 + yield 或 array_walk_recursive(注意它跳过关联键,仅遍历叶子值);更稳妥用自定义递归函数配合 isset;
  • 求两个数组交集/差集(含键名):区分 array_intersect(比对值)、array_intersect_key(比对键)、array_intersect_assoc(键+值都等);
  • 扁平化嵌套数组:PHP 7.4+ 推荐 array_reduce($arr, fn($carry, $item) => array_merge($carry, is_array($item) ? $item : [$item]), []);兼容旧版可用 array_walk_recursive 收集。

性能与内存关键点

大数组处理时,细节决定响应速度与 OOM 风险:

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

  • 避免在循环中反复 count($arr):PHP 7.2+ 已优化,但低版本仍建议提前缓存 $len = count($arr)
  • 使用 range() + array_flip() 构建映射表比循环赋值快数倍,尤其用于白名单校验(如 $validTypes = array_flip(['jpg','png','gif']); if (isset($validTypes[$ext])) {...});
  • unset() 删除元素不会释放内存空间,只是标记为“已删除”,后续插入可能复用该槽位;真正收缩需 $arr = array_values($arr)(重建索引)或 $arr = array_filter($arr, function($v){return $v !== null;})(过滤后重索引);
  • 大量小数组拼接,用 array_merge 比 [] + [] 效率低,因前者每次调用都复制整个数组;推荐先收集再一次性合并。
text=ZqhQzanResources