コールバック関数は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)
}
PHP 5.6では、値の代わりにキーでフィルタリングするように設定できるarray_filter()
,に 3 番目のパラメータが導入されました。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
);
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
して、値とキーの両方をフィルター関数に渡すこともできます。最初の例に基づいた不自然な例を次に示しますが、$allowed
この方法を使用してフィルタリング ルールをエンコードすることはお勧めしません。
$my_array = ['foo' => 1, 'bar' => 'baz', 'hello' => 'wld'];
$allowed = ['foo' => true, 'bar' => true, 'hello' => 'world'];
$filtered = array_filter(
$my_array,
// N.b. it's ($val, $key) not ($key, $val):
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)
}
したがって、関数では、他の特定のテストを実行できます。
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_USE_KEY
フラグを使用できarray_filter
ます。
$result = array_filter($my_array, function ($k) use ($allowed) {
return in_array($k, $allowed);
}, ARRAY_FILTER_USE_KEY);
それ以外の場合は、次の関数を使用できます ( 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;
}
}
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
)
これは、後で使用するためにフィルター処理された値を保持したいが、そうでないことが確実な場合は、より整理したい場合には適用できません。
@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 より小さいと仮定すると、後者の方が効率的です。
一度だけ必要な場合はやり過ぎかもしれませんが、YaLinqoライブラリー* を使用してコレクションをフィルタリング (およびその他の変換を実行) できます。このライブラリを使用すると、流暢な構文でオブジェクトに対して SQL に似たクエリを実行できます。そのwhere
関数は、値とキーの 2 つの引数を持つカルバックを受け入れます。例えば:
$filtered = from($array)
->where(function ($v, $k) use ($allowed) {
return in_array($k, $allowed);
})
->toArray();
(関数は反復子を返すため、結果のシーケンスを 1 回where
だけ反復処理する必要がある場合は削除できます。)foreach
->toArray()
※私が開発した