应先用 array_values 确保数字索引,再用 array_filter 配合 ARRAY_FILTER_USE_BOTH 和 is_int($k) && $k % 2 === 0 筛选偶数位置元素,最后按需调用 array_values 重置键。

php array_merge 合并后怎么只保留偶数索引?
直接用 array_merge 会重排索引,原始键名(尤其是字符串键)全丢,数字键也从 0 开始连续编号——这和“保留偶数索引”完全冲突。真要保留原数组的偶数位置(比如第 0、2、4…个子数组),得先明确:你指的是「按顺序位置取」,不是「键名是偶数」。
常见误操作:array_merge(...$arr) 后再用 array_filter 检查键,结果发现键全变成 0,1,2…了,根本没偶数可筛。
- 如果源数组是
[[a,b], [c,d], [e,f], [g,h]],你要的是索引 0 和 2 的子数组 →[[a,b], [e,f]] - 如果源数组带字符串键如
['x'=>[a], 'y'=>[b], 'z'=>[c]],它没有数字索引 0/2,“偶数索引”无意义,需先用array_values强制转为数字索引 - 真正该用的是
array_filter配合ARRAY_FILTER_USE_KEY,但前提是键没被重排过
用 array_filter + 键回调提取偶数位置子数组
核心是遍历原二维数组,用当前数字键判断是否为偶数。注意:必须确保键是整型数字,否则 $k % 2 === 0 可能出错(比如键是字符串 "0" 或浮点 2.0)。
$arr = [[1,2], [3,4], [5,6], [7,8], [9,10]];
$result = array_filter($arr, function($v, $k) {
return is_int($k) && $k % 2 === 0;
}, ARRAY_FILTER_USE_BOTH);
// $result 是 [0=>[1,2], 2=>[5,6], 4=>[9,10]],键保留- 必须加
is_int($k),避免键是字符串时"2" % 2返回 0(PHP 会静默转数字) - 若要重置键为 0,1,2…,最后补上
array_values($result) - 别用
foreach手动 push,array_filter更简洁且保持原键语义
遇到关联键或乱序键怎么办?
如果原始二维数组是 ['a'=>[1], 'c'=>[2], 'e'=>[3], 'g'=>[4]],它的键不是数字,$k % 2 没法判断“第几个”。这时得先用 array_values 强制提取值并生成连续数字键,再筛偶数位置:
立即学习“PHP免费学习笔记(深入)”;
$arr = ['a'=>[1], 'c'=>[2], 'e'=>[3], 'g'=>[4]]; $indexed = array_values($arr); // → [[1],[2],[3],[4]] $result = array_filter($indexed, fn($v,$k) => is_int($k) && $k % 2 === 0, ARRAY_FILTER_USE_BOTH); // 得到 [0=>[1], 2=>[3]]
-
array_values不改变值,只重置键,是安全的中间步骤 - 如果原数组键很重要(比如要回溯来源),就不能丢,得自己计数:
$i=0; foreach($arr as $k=>$v){ if($i++ % 2 === 0) $out[$k] = $v; } - 别试图用
array_keys再映射回去——多此一举且易错
性能敏感场景下别用 array_filter
对超大二维数组(比如上万项),array_filter 会遍历全部元素。如果只取前几个偶数位置,用 for 加步长更省:
$arr = /* huge array */;
$result = [];
for ($i = 0; $i < count($arr); $i += 2) {
if (isset($arr[$i])) {
$result[] = $arr[$i];
}
}- 比
array_filter少一半迭代次数,且不生成临时键值对 - 仅适用于纯数字索引且连续的情况;有空洞(如键是 0,2,4 但缺 6)时,
count和isset必须配合 - 如果数组稀疏(键分散),还是老实用
array_keys+foreach遍历键更稳
实际用哪一种,取决于你手上的数组到底长什么样——键是数字还是字符串、是否连续、要不要保留原键、数据量多大。漏掉这些前提直接套函数,十有八九得到的不是你想要的“偶数索引”。










