array_filter()中的回调函数只传递数组的值,而不是键。
如果我有:
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
删除$my_array中不在$allowed数组中的所有键的最佳方法是什么?
期望的输出:
$my_array = array("foo" => 1);
array_filter()中的回调函数只传递数组的值,而不是键。
如果我有:
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
删除$my_array中不在$allowed数组中的所有键的最佳方法是什么?
期望的输出:
$my_array = array("foo" => 1);
当前回答
天真而丑陋(但似乎更快)的解决方案?
只在php 7.3.11中尝试了这个,但是一个丑陋的循环似乎在大约三分之一的时间内执行。在一个有几百个键的数组上也有类似的结果。微优化,可能在RW中没有用,但发现它令人惊讶和有趣:
$time = microtime(true);
$i = 100000;
while($i) {
$my_array = ['foo' => 1, 'hello' => 'world'];
$allowed = ['foo', 'bar'];
$filtered = array_filter(
$my_array,
function ($key) use ($allowed) {
return in_array($key, $allowed);
},
ARRAY_FILTER_USE_KEY
);
$i--;
}
print_r($filtered);
echo microtime(true) - $time . ' on array_filter';
// 0.40600109100342 on array_filter
$time2 = microtime(true);
$i2 = 100000;
while($i2) {
$my_array2 = ['foo' => 1, 'hello' => 'world'];
$allowed2 = ['foo', 'bar'];
$filtered2 = [];
foreach ($my_array2 as $k => $v) {
if (in_array($k, $allowed2)) $filtered2[$k] = $v;
}
$i2--;
}
print_r($filtered2);
echo microtime(true) - $time2 . ' on ugly loop';
// 0.15677785873413 on ugly loop
其他回答
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
);
由于PHP 7.4引入了箭头函数,我们可以让它更简洁:
$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将值和键同时传递给筛选函数。下面是一个基于第一个例子的例子,但是注意,我不建议使用$允许这种方式编码过滤规则:
$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']
下面是一个使用闭包的更灵活的解决方案:
$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)
}
在函数中,你可以做其他特定的测试。
我使用了一个小的“Utils”类,其中我添加了两个过滤器静态函数来使用denylist或allowlist来过滤数组。
<?php
class Utils {
/**
* Filter an array based on a allowlist of keys
*
* @param array $array
* @param array $allowlist
*
* @return array
*/
public static function array_keys_allowlist( array $array, array $allowlist ): array {
return array_intersect_key( $array, array_flip( $allowlist ) );
}
/**
* Filter an array based on a denylist of keys
*
* @param array $array
* @param array $denylist
*
* @return array
*/
public static function array_keys_denylist( array $array, array $denylist ): array {
return array_diff_key($array,array_flip($denylist));
}
}
然后你可以像这样使用它
<?php
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
$my_array = Utils::array_keys_allowlist($my_array, $allowed)
如果你正在寻找一个方法,通过出现在键中的字符串来过滤数组,你可以使用:
$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));
print_r($mResult)的结果是
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
)
如果只需要一次,可能会有些多余,但您可以使用YaLinqo库*来筛选集合(并执行任何其他转换)。这个库允许以流畅的语法对对象执行类似sql的查询。在这里,函数接受带有两个参数的回调:一个值和一个键。例如:
$filtered = from($array)
->where(function ($v, $k) use ($allowed) {
return in_array($k, $allowed);
})
->toArray();
(where函数返回一个迭代器,所以如果你只需要在结果序列上用foreach迭代一次,->toArray()可以被删除。)
*由我开发