array_filter()
中的回调函数仅传递数组值,而不是键。
如果我有:
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
如何删除$my_array
中所有不在$allowed
数组中的键?
期望输出:
$my_array = array("foo" => 1);
使用array_intersect_key
和array_flip
:
var_dump(array_intersect_key($my_array, array_flip($allowed)));
array(1) {
["foo"]=>
int(1)
}
array_flip
?只需使用键定义 $allowed
:
allowed = array ( 'foo' => 1, 'bar' => 1 );
- Yuval A.PHP 5.6引入了第三个参数到array_filter()
函数中,flag
,你可以将其设置为ARRAY_FILTER_USE_KEY
以按键过滤而不是按值过滤:
$my_array = ['foo' => 1, 'hello' => 'world'];
$allowed = ['foo', 'bar'];
$filtered = array_filter(
$my_array,
function ($key) use ($allowed) {
// N.b. in_array() is notorious for being slow
return in_array($key, $allowed);
},
ARRAY_FILTER_USE_KEY
);
$my_array = ['foo' => 1, 'hello' => 'world'];
$allowed = ['foo', 'bar'];
$filtered = array_filter(
$my_array,
fn ($key) => in_array($key, $allowed),
ARRAY_FILTER_USE_KEY
);
显然这种方法不如array_intersect_key($my_array, array_flip($allowed))
优雅,但它提供了额外的灵活性,可以对键执行任意测试,例如$allowed
可以包含正则表达式模式而不是普通字符串。
您还可以使用ARRAY_FILTER_USE_BOTH
将值和键都传递给过滤函数。以下是一个基于第一个示例的人为示例,但请注意,我不建议使用$allowed
来编码过滤规则:
$my_array = ['foo' => 1, 'bar' => 'baz', 'hello' => 'wld'];
$allowed = ['foo' => true, 'bar' => true, 'hello' => 'world'];
$filtered = array_filter(
$my_array,
fn ($val, $key) => isset($allowed[$key]) && (
$allowed[$key] === true || $allowed[$key] === $val
),
ARRAY_FILTER_USE_BOTH
); // ['foo' => 1, 'bar' => 'baz']
$filtered = array_filter(
$my_array,
fn ($key) => in_array($key, $allowed),
ARRAY_FILTER_USE_KEY
);
- jartaudin_array()
的迭代调用的答案都不会比更优雅的array_intersect_key()
调用更有效。是的,查找数组需要翻转一次,但由于PHP非常快速地进行键查找(例如isset()
),我预计在大多数测试用例中in_array()
将被抛在后面。更简单地说,已经证明isset()
在基准测试中远远优于in_array()
。唯一需要注意的危险是当翻转技术改变值时--例如当您将浮点值翻转为键时,它变成了整数。 - mickmackusa这里是一个使用闭包的更灵活的解决方案:
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
$result = array_flip(array_filter(array_flip($my_array), function ($key) use ($allowed)
{
return in_array($key, $allowed);
}));
var_dump($result);
输出:
array(1) {
'foo' =>
int(1)
}
因此在这个函数中,您可以进行其他特定的测试。
array_flip($my_array)
是危险的。如果数组中存在重复值,则数组的大小将会减小,因为在同一级别上数组不能有重复的键。这种方法不应该被使用——它是不稳定/不可靠的。 - mickmackusaarray_filter
时获取数组的当前键尽管我喜欢Vincent提供的解决方案,但它实际上并没有使用array_filter
。如果您从搜索引擎进入此处,并且想要在array_filter
的回调函数中访问当前迭代的键,您可能正在寻找类似于以下内容的解决方案(PHP >= 5.3):
$my_array = ["foo" => 1, "hello" => "world"];
$allowed = ["foo", "bar"];
reset($my_array ); // Unnecessary in this case, as we just defined the array, but
// make sure your array is reset (see below for further explanation).
$my_array = array_filter($my_array, function($value) use (&$my_array, $allowed) {
$key = key($my_array); // request key of current internal array pointer
next($my_array); // advance internal array pointer
return isset($allowed[$key]);
});
// $my_array now equals ['foo' => 1]
它将你要过滤的数组作为引用传递给回调函数。因为array_filter
不会按照增加其公共内部指针的方式惯常迭代数组,所以你需要自己进行指针的移动。
重要的是,你需要确保重新设置数组,否则你可能会从中间开始(因为内部数组指针被你之前执行的某些代码留在那里)。
$&array
不是有效的 PHP 代码,并且 each()
已经自 PHP7.2 开始废弃,并在 PHP8 中被彻底移除。 - mickmackusaeach()
也没有被弃用。在我看来,我的回答的要点可以轻松地转移到提问者的问题上,但我已经相应地更新了它,现在可以直接复制粘贴而无需过多考虑。我还修正了引用中的小错别字($&
=> &$
)。如果您对我的回答仍有不满意之处,请随意编辑。干杯! - fluARRAY_FILTER_USE_KEY
标志并不常见。所有在SO上的答案都是时代的产物,可能在半个多十年后已经无效、不准确或者不再有帮助了。我其实不知道现在已经弃用的答案是否应该因为历史原因而保留或删除。有人可能仍然被迫支持使用已经过时的PHP版本的项目。 - fluARRAY_FILTER_USE_KEY
标志时,是否有一种方法“使用 array_filter() 过滤数组键?”这将非常有趣。像这样的问题有很多,它们提出了一个挑战:这真的可能吗?由于当时没有任何答案实际使用 array_filter
,所以我认为我的答案仍然有价值。 - flu$array = array(
1 => 'one',
2 => 'two',
3 => 'three'
);
$disallowed = array(1,3);
foreach($disallowed as $key){
unset($array[$key]);
}
print_r($array)
的结果为:
Array
(
[2] => two
)
如果您想将过滤后的值保留以供以后使用,但更加整洁,那么这不适用于您,如果您确定您不需要这样做。
unset()
函数时,无需检查数组键是否存在,即使键不存在也不会发出警告。 - Christopher参照@sepiariver的建议,我在PHP 8.0.3上进行了类似的测试:
$arr = ['a' => 1, 'b' => 2, 'c' => 3, 'd' => 4, 'e' => 5, 'f' => 6, 'g' => 7, 'h' => 8];
$filter = ['a', 'e', 'h'];
$filtered = [];
$time = microtime(true);
$i = 1000000;
while($i) {
$filtered = array_intersect_key($arr, array_flip($filter));
$i--;
}
print_r($filtered);
echo microtime(true) - $time . " using array_intersect_key\n\n";
$filtered = [];
$time = microtime(true);
$i = 1000000;
while($i) {
$filtered = array_filter(
$arr,
function ($key) use ($filter){return in_array($key, $filter);},
ARRAY_FILTER_USE_KEY
);
$i--;
}
print_r($filtered);
echo microtime(true) - $time . " using array_filter\n\n";
$filtered = [];
$time = microtime(true);
$i = 1000000;
while($i) {
foreach ($filter as $key)
if(array_key_exists($key, $arr))
$filtered[$key] = $arr[$key];
$i--;
}
print_r($filtered);
echo microtime(true) - $time . " using foreach + array_key_exists\n\n";
array_filter的问题在于它会遍历$arr的所有元素,而array_intersect_key和foreach只会遍历$filter。假设$filter比$arr小,则后两种方法更有效率。
如果你正在寻找一种通过键值中的字符串过滤数组的方法,你可以使用以下代码:
$mArray=array('foo'=>'bar','foo2'=>'bar2','fooToo'=>'bar3','baz'=>'nope');
$mSearch='foo';
$allowed=array_filter(
array_keys($mArray),
function($key) use ($mSearch){
return stristr($key,$mSearch);
});
$mResult=array_intersect_key($mArray,array_flip($allowed));
< p > print_r($mResult) 的结果为 < /p >
Array ( [foo] => bar [foo2] => bar2 [fooToo] => bar3 )
这是支持正则表达式的答案的改编
function array_preg_filter_keys($arr, $regexp) {
$keys = array_keys($arr);
$match = array_filter($keys, function($k) use($regexp) {
return preg_match($regexp, $k) === 1;
});
return array_intersect_key($arr, array_flip($match));
}
$mArray = array('foo'=>'yes', 'foo2'=>'yes', 'FooToo'=>'yes', 'baz'=>'nope');
print_r(array_preg_filter_keys($mArray, "/^foo/i"));
输出
Array
(
[foo] => yes
[foo2] => yes
[FooToo] => yes
)
stristr
会对最终用户做出一些假设。也许允许用户传递一个正则表达式会更好,这将使他们在锚点、单词边界和大小写敏感等方面拥有更多的灵活性。 - mačekPHP中的数组过滤函数:
array_filter ( $array, $callback_function, $flag )
$array - 这是输入的数组。
$callback_function - 要使用的回调函数,如果回调函数返回 true ,则从数组中返回当前值到结果数组。
$flag - 它是可选参数,它将确定发送到回调函数的参数。如果此参数为空,则回调函数将采用数组值作为参数。如果你想把数组键作为参数发送,那么应该使用$flag作为 ARRAY_FILTER_USE_KEY。如果你想发送键和值,那么应该使用$flag作为 ARRAY_FILTER_USE_BOTH 。
例如:考虑一个简单的数组
$array = array("a"=>1, "b"=>2, "c"=>3, "d"=>4, "e"=>5);
如果您想根据数组键(array key)筛选数组,则需要在array_filter的第三个参数中使用ARRAY_FILTER_USE_KEY函数。
$get_key_res = array_filter($array,"get_key",ARRAY_FILTER_USE_KEY );
如果您想根据数组键和数组值过滤数组,则需要在array_filter函数的第三个参数中使用ARRAY_FILTER_USE_BOTH。
$get_both = array_filter($array,"get_both",ARRAY_FILTER_USE_BOTH );
示例回调函数:
function get_key($key)
{
if($key == 'a')
{
return true;
} else {
return false;
}
}
function get_both($val,$key)
{
if($key == 'a' && $val == 1)
{
return true;
} else {
return false;
}
}
Output of $get_key is :Array ( [a] => 1 )
Output of $get_both is :Array ( [a] => 1 )
从PHP 5.6开始,您可以在array_filter
中使用ARRAY_FILTER_USE_KEY
标志:
$result = array_filter($my_array, function ($k) use ($allowed) {
return in_array($k, $allowed);
}, ARRAY_FILTER_USE_KEY);
function filter_array_keys(array $array, $callback)
{
$matchedKeys = array_filter(array_keys($array), $callback);
return array_intersect_key($array, array_flip($matchedKeys));
}
$result = filter_array_keys($my_array, function ($k) use ($allowed) {
return in_array($k, $allowed);
});
function filter_array_keys(array $array, $keys)
{
if (is_callable($keys)) {
$keys = array_filter(array_keys($array), $keys);
}
return array_intersect_key($array, array_flip($keys));
}
// using a callback, like array_filter:
$result = filter_array_keys($my_array, function ($k) use ($allowed) {
return in_array($k, $allowed);
});
// or, if you already have the keys:
$result = filter_array_keys($my_array, $allowed));
最后但并非最不重要的,您也可以使用简单的foreach
:
$result = [];
foreach ($my_array as $key => $value) {
if (in_array($key, $allowed)) {
$result[$key] = $value;
}
}
in_array()
是PHP中性能最差的数组搜索函数之一。添加库的开销只会进一步降低性能。由于两个本地函数或循环unset调用可以简洁地解决此问题,因此我永远不会使用库的方法。 - mickmackusaARRAY_FILTER_USE_KEY
的 array_filter
),使用不寻常的其他语言没有的函数(array_flip
) 或深入了解PHP架构(知道 unset
具有哈希表访问速度和 in_array
线性扩展)。 - Athari
$b = ['foo' => $a['foo'], 'bar' => $a['bar']]
。这将导致$b['bar']
为null
。 - oriadam