繁体   English   中英

如何使用 PHP 中的索引数组的键过滤关联的 arrays?

[英]How to filter an associative arrays using keys of an indexed array in PHP?

array_filter()中的回调 function 只传入数组的值,而不是键。

如果我有:

$my_array = array("foo" => 1, "hello" => "world");

$allowed = array("foo", "bar");

删除$my_array中不在$allowed数组中的所有键的最佳方法是什么?

所需的 output:

$my_array = array("foo" => 1);

使用array_intersect_keyarray_flip

var_dump(array_intersect_key($my_array, array_flip($allowed)));

array(1) {
  ["foo"]=>
  int(1)
}

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将值和键都传递给您的过滤器 function。 这是一个基于第一个的人为示例,但请注意,我不建议以这种方式使用$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']

我需要做同样的事情,但在键上使用更复杂的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
)

这是使用闭包的更灵活的解决方案:

$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)
}

所以在function中,可以做其他具体的测试。

这是使用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
)

如果您想保留过滤后的值以供以后使用但更整洁(如果您确定不这样做),则此方法不适用。

如果您正在寻找一种通过键中出现的字符串过滤数组的方法,您可以使用:

$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"));

Output

Array
(
    [foo] => yes
    [foo2] => yes
    [FooToo] => yes
)

使用array_filter时如何获取数组的当前键

不管我多么喜欢文森特对马切克问题的解决方案,它实际上并没有使用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通常不会通过增加它的公共内部指针来迭代数组,因此您必须自己推进它。

这里重要的是你需要确保你的数组被重置,否则你可能会从它的中间开始(因为内部数组指针被你之前执行的一些代码留在那里)。

从 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 ( 来自 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;
    }
}

基于@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";
  • 0.28603601455688 使用 array_intersect_key
  • 1.3096671104431 使用 array_filter
  • 0.19402384757996 使用 foreach + array_key_exists

array_filter 的“问题”是它将循环遍历 $arr 的所有元素,而 array_intersect_key 和 foreach 仅循环遍历 $filter。 后者更有效,假设 $filter 小于 $arr。

来自 php 的阵列过滤器 function:

array_filter ( $array, $callback_function, $flag )

$array - 它是输入数组

$callback_function - 使用回调 function, 如果回调 function 返回true ,则将数组中的当前值返回到结果数组中。

$flag - 它是可选参数,它将确定发送什么 arguments 到回调 function。 如果此参数为空,则回调 function 将数组值作为参数。 如果要发送数组键作为参数,则使用 $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作为数组 function array_filter 的第三个参数

$get_key_res = array_filter($array,"get_key",ARRAY_FILTER_USE_KEY );

如果要根据数组键和数组值过滤数组,我们需要使用ARRAY_FILTER_USE_BOTH作为数组 function 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

Output of $get_key is :Array ( [a] => 1 ) 
Output of $get_both is :Array ( [a] => 1 ) 

如果您只需要一次,这可能有点过头了,但您可以使用YaLinqo库* 过滤 collections(并执行任何其他转换)。 该库允许对具有流畅语法的对象执行类似 SQL 的查询。 where function 接受带有两个 arguments 的回调:一个值和一个键。 例如:

$filtered = from($array)
    ->where(function ($v, $k) use ($allowed) {
        return in_array($k, $allowed);
    })
    ->toArray();

where function 返回一个迭代器,因此如果您只需要使用foreach对结果序列进行一次迭代,则可以删除->toArray() 。)

* 由我开发

使用此功能,您可以过滤多维数组

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];

}

// 过滤掉键短于 4 个字符的数组元素 // 通过使用匿名函数和闭包...

function comparison($min)
{
   return function($item) use ($min) { 
      return strlen($item) >= $min;   
   }; 
}

$input = array(
  0      => "val 0",
  "one"  => "val one",
  "two"  => "val two",
  "three"=> "val three",
  "four" => "val four",  
  "five" => "val five",    
  "6"    => "val 6"    
);

$output = array_filter(array_keys($input), 比较(4));

打印_r($输出);
在此处输入图片说明

天真而丑陋(但似乎更快)的解决方案?

仅在 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

为了达到相反的效果 - 删除数组键。

$my_array = array("foo" => 1, "hello" => "world");
$not_allowed = array("foo", "bar");

期望的输出:

$my_array = array("hello" => "world");

跑:

$my_array = array_diff_key($my_array, array_flip($not_allowed));

也适用于数字索引数组 - 字符串和整数。

我使用一个小的“Utils”class,在其中添加两个过滤器 static function 使用黑名单或白名单过滤数组。

<?php

class Utils {
 
  /**
   * Filter an array based on a white list of keys
   *
   * @param array $array
   * @param array $whitelist
   *
   * @return array
   */
  public static function array_keys_whitelist( array $array, array $whitelist ): array {
    return array_intersect_key( $array, array_flip( $whitelist ) );
  }


  /**
   * Filter an array based on a black list of keys
   *
   * @param array $array
   * @param array $blacklist
   *
   * @return array
   */
  public static function array_keys_blacklist( array $array, array $blacklist ): array {
    return array_diff_key($array,array_flip($blacklist));
  }

}

然后你可以像这样使用它

<?php

$my_array = array("foo" => 1, "hello" => "world");
$allowed = array("foo", "bar");

$my_array = Utils::array_keys_whitelist($my_array,  $allowed)
$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 )

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM