unset()删除元素不重排键名,数字索引数组易致for循环越界或漏项;适用关联数组或无需连续索引场景;批量删除需先收集键名再unset,避免foreach中直接修改。

unset() 删除后键名不重排,遍历时可能出错
直接用 unset() 删数组元素最常用,但它只解除键与值的绑定,不会压缩索引——尤其对数字索引数组,删掉中间某项后,foreach 没问题,但用 for 配 count() 就容易越界或漏项。
- 适用场景:明确知道键名,且后续不依赖连续数字索引(比如关联数组)
- 错误现象:
for ($i = 0; $i 删了 <code>$arr[1]后,$arr[2]变成新“第2个”,但$i还是按 0→1→2 走,会访问空值或报 Notice - 补救方法:删完立刻用
array_values($arr)重排键,但注意这会把所有键转成数字索引,丢掉原关联键
array_splice() 适合按位置删除,自动重排索引
array_splice() 是真正“切掉一段”的操作,它会修改原数组、删除指定位置和长度的元素,并自动让后续数字键前移。比 unset() + array_values() 组合更干净,尤其处理索引数组时。
- 参数差异:
array_splice($arr, $offset, $Length)——$offset从 0 开始计,负数从尾部算;$length省略则删到末尾 - 注意点:它会**直接修改原数组**,不是返回新数组(除非你用变量接收返回值,但原数组已变)
- 性能影响:对大数组,重排索引有轻微开销,但通常可忽略;比
unset()后再array_values()少一次遍历 - 示例:
array_splice($arr, 2, 1)删掉第 3 个元素(索引 2),后面所有数字键减 1
array_filter() 删除满足条件的多个元素,不破坏键名
要根据值内容批量删(比如删掉所有空字符串、所有小于 10 的数),array_filter() 最合适。它不改变原数组,返回过滤后的新数组,键名默认保留——这对关联数组很友好。
- 常见错误:忘了返回
true或false,导致全删或全留;匿名函数里没加return是常见坑 - 使用场景:去重、清空假值(
array_filter($arr)默认删掉0、''、NULL、false)、按业务逻辑筛选 - 兼容性:php 5.6+ 支持闭包,早期版本需用
create_function()(不推荐)或提前定义函数 - 如果需要重排数字键,仍得套一层
array_values()
unset() 在 foreach 中直接删会报错
在 foreach 遍历数组时,边循环边用 unset() 删当前项,PHP 会触发 “Cannot unset String offsets” 或实际执行异常——不是语法错,而是内部指针和结构不一致导致行为不可靠。
立即学习“PHP免费学习笔记(深入)”;
- 根本原因:PHP 的
foreach底层用哈希表迭代器,unset()修改结构后,迭代器可能指向已释放内存 - 安全做法:先收集要删的键,循环结束再统一
unset();或者改用for+array_keys()倒序删(避免键位偏移) - 示例避坑:
$keys_to_remove = []; foreach ($arr as $k => $v) { if ($v === 'bad') $keys_to_remove[] = $k; } foreach ($keys_to_remove as $k) unset($arr[$k]);
删数组不是选函数那么简单,关键是看你要不要保留键名、是否按值判断、是否在循环中操作——这几个条件一组合,unset()、array_splice()、array_filter() 的适用边界就非常清晰。最容易被跳过的,是删完之后要不要重排索引,以及 foreach 里动手脚的风险。