array_filter()
中的回调函数只传入数组的值,而不是键。
如果我有:
$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");
删除$my_array
数组中不在$allowed
数组中的所有键的最佳方法是什么?
期望的输出:
$my_array = array("foo" => 1);
答案 0 :(得分:431)
array_intersect_key
和array_flip
:
var_dump(array_intersect_key($my_array, array_flip($allowed)));
array(1) {
["foo"]=>
int(1)
}
答案 1 :(得分:252)
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) {
return 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,
function ($val, $key) use ($allowed) { // N.b. $val, $key not $key, $val
return isset($allowed[$key]) && (
$allowed[$key] === true || $allowed[$key] === $val
);
},
ARRAY_FILTER_USE_BOTH
); // ['foo' => 1, 'bar' => 'baz']
答案 2 :(得分:39)
我需要这样做,但键上有一个更复杂的array_filter
。
以下是我使用类似方法的方法。
// Filter out array elements with keys shorter than 4 characters
$a = array(
0 => "val 0",
"one" => "val one",
"two" => "val two",
"three"=> "val three",
"four" => "val four",
"five" => "val five",
"6" => "val 6"
);
$f = array_filter(array_keys($a), function ($k){ return strlen($k)>=4; });
$b = array_intersect_key($a, array_flip($f));
print_r($b);
输出结果:
Array
(
[three] => val three
[four] => val four
[five] => val five
)
答案 3 :(得分:8)
这是一个使用闭包的更灵活的解决方案:
$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)
}
因此在函数中,您可以进行其他特定测试。
答案 4 :(得分:4)
如果您正在寻找通过键中出现的字符串过滤数组的方法,可以使用:
$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
)
答案 5 :(得分:4)
array_filter
无论我如何看待Vincent对Maček问题的解决方案,它实际上并没有使用array_filter
。如果你从搜索引擎来到这里,你可能会在这里寻找类似的东西( PHP> = 5.3 ):
$array = ['apple' => 'red', 'pear' => 'green'];
reset($array); // Unimportant here, but make sure your array is reset
$apples = array_filter($array, function($color) use ($&array) {
$key = key($array);
next($array); // advance array pointer
return key($array) === 'apple';
}
它传递您要过滤的数组作为回调的引用。由于array_filter
通常不会通过增加它的公共内部指针来迭代数组,所以你必须自己推进它。
这里重要的是你需要确保你的阵列被重置,否则你可能会在它的中间开始。
在 PHP> = 5.4 中,您可以让回调更短:
$apples = array_filter($array, function($color) use ($&array) {
return each($array)['key'] === 'apple';
}
答案 6 :(得分:3)
这是使用unset()的不太灵活的替代方案:
$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
)
如果您希望保留过滤的值以供日后使用,但如果您确定不这样做,则不适用。
答案 7 :(得分:3)
从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);
否则,您可以使用此功能(from TestDummy):
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;
}
}
答案 8 :(得分:2)
基于@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_filter 和 foreach 只遍历 $filter。后者更有效,假设 $filter 小于 $arr。
答案 9 :(得分:1)
如果您只需要一次就可能是一种矫枉过正,但您可以使用YaLinqo库*来过滤集合(并执行任何其他转换)。该库允许使用流畅的语法对对象执行类似SQL的查询。它的where
函数接受一个带有两个参数的回调:一个值和一个键。例如:
$filtered = from($array)
->where(function ($v, $k) use ($allowed) {
return in_array($k, $allowed);
})
->toArray();
(where
函数返回一个迭代器,因此如果您只需要在结果序列上使用foreach
进行一次迭代,就可以删除->toArray()
。)
*由我开发
答案 10 :(得分:0)
使用此功能,您可以过滤多维数组
function filter_array_keys($array,$filter_keys=array()){
$l=array(&$array);
$c=1;
//This first loop will loop until the count var is stable//
for($r=0;$r<$c;$r++){
//This loop will loop thru the child element list//
$keys = array_keys($l[$r]);
for($z=0;$z<count($l[$r]);$z++){
$object = &$l[$r][$keys[$z]];
if(is_array($object)){
$i=0;
$keys_on_array=array_keys($object);
$object=array_filter($object,function($el) use(&$i,$keys_on_array,$filter_keys){
$key = $keys_on_array[$i];
$i++;
if(in_array($key,$filter_keys) || is_int($key))return false;
return true;
});
}
if(is_array($l[$r][$keys[$z]])){
$l[] = &$l[$r][$keys[$z]];
$c++;
}//IF
}//FOR
}//FOR
return $l[0];
}
答案 11 :(得分:0)
数组过滤函数:
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_FILTER_USE_KEY 作为数组函数array_filter的第三个参数。< / p>
$get_key_res = array_filter($array,"get_key",ARRAY_FILTER_USE_KEY );
如果要根据数组键和数组值过滤数组,我们需要使用 ARRAY_FILTER_USE_BOTH 作为数组函数array_filter的第三个参数。
$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 )
答案 12 :(得分:0)
//使用短于4个字符的键过滤掉数组元素 //使用Anonymous函数和Closure ...
<p id="changeElem"><span id="output" style="font-weight: bold" >Before change</span></p>
$ output = array_filter(array_keys($ input),comparison(4));
答案 13 :(得分:0)
天真丑陋(但似乎更快)的解决方案?
仅在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
答案 14 :(得分:-1)
$elements_array = ['first', 'second'];
删除某些数组元素的功能
function remove($arr, $data) {
return array_filter($arr, function ($element) use ($data) {
return $element != $data;
});
}
致电并打印
print_r(remove($elements_array, 'second'));
结果
Array ( [0] => first )