PHP - получить все ключи из массива, которые начинаются с определенной строки

84

У меня есть массив, который выглядит так:

array(
  'abc' => 0,
  'foo-bcd' => 1,
  'foo-def' => 1,
  'foo-xyz' => 0,
  // ...
)

Как я могу получить только те элементы, с которых начинается foo-?

Alex
источник
3
Наличие массива с такими префиксами - это запах кода. Извлеките эти значения в объект, содержащий эти значения, или, по крайней мере, сделайте префикс, указывающий на массив, и добавьте к нему значения.
Гордон
См. Также: stackoverflow.com/questions/13766898/…
dreftymac

Ответы:

20
$arr_main_array = array('foo-test' => 123, 'other-test' => 456, 'foo-result' => 789);

foreach($arr_main_array as $key => $value){
    $exp_key = explode('-', $key);
    if($exp_key[0] == 'foo'){
         $arr_result[] = $value;
    }
}

if(isset($arr_result)){
    print_r($arr_result);
}
Разработчик-Сид
источник
18
Прежде чем принять это решение, лучше использовать тот, что ниже. Никогда не используйте функции массива, где достаточно простой строковой функции. Строковые функции НАМНОГО быстрее. Это будет "отставать" с большими массивами.
Martijn
2
Я не вижу смысла разбивать ключи массива на массивы. Это строки, и исходный вопрос просто касается префикса строки. Также OP запрашивает «элементы», а не только значения, поэтому результат с числовой индексацией будет бесполезен - здесь явно важны ключи, а значения (предположительно) являются анонимными флагами 1/0.
Джейсон
127

Функциональный подход:

Выберите какую- array_filter_keyто функцию из комментариев в http://php.net/array_filter или напишите свою. Тогда вы могли:

$array = array_filter_key($array, function($key) {
    return strpos($key, 'foo-') === 0;
});

Процедурный подход:

$only_foo = array();
foreach ($array as $key => $value) {
    if (strpos($key, 'foo-') === 0) {
        $only_foo[$key] = $value;
    }
}

Процедурный подход с использованием объектов:

$i = new ArrayIterator($array);
$only_foo = array();
while ($i->valid()) {
    if (strpos($i->key(), 'foo-') === 0) {
        $only_foo[$i->key()] = $i->current();
    }
    $i->next();
}
Эриско
источник
20
Примечание. PHP 5.6 добавляет к нему флаг ARRAY_FILTER_USE_KEY array_filter(), который можно использовать вместо предоставления пользовательской array_filter_key()функции.
Джейсон
4
Не могли бы вы просто добавить в свой пример функцию array_filter_key? Я все время получал ошибку, и мне пришлось перечитывать ваш пост несколько раз, прежде чем я осознал свою ошибку. Нетрудно просто опубликовать код, сохранить щелчок и сэкономить время на устранение неполадок. Благодарю.
Крис Спрэг
38

Я бы сделал это так, хотя не могу дать вам более эффективного совета, прежде чем поймете, что вы хотите делать с полученными вами ценностями.

$search = "foo-";
$search_length = strlen($search);
foreach ($array as $key => $value) {
    if (substr($key, 0, $search_length) == $search) {
        ...use the $value...
    }
}
Матиас Канепа
источник
2
Вы также можете использовать:foreach ($array as $key => $value) { if (0 === strpos($key, 'foo-')) { ...use the $value... } }
Яго
«4» здесь нужно отрегулировать на любую длину «foo-».
Джейсон
22

Просто я использовал array_filterфункцию для достижения решения следующим образом

<?php

$input = array(
    'abc' => 0,
    'foo-bcd' => 1,
    'foo-def' => 1,
    'foo-xyz' => 0,
);

$filtered = array_filter($input, function ($key) {
    return strpos($key, 'foo-') === 0;
}, ARRAY_FILTER_USE_KEY);

print_r($filtered);

Вывод

Array
(
    [foo-bcd] => 1
    [foo-def] => 1
    [foo-xyz] => 0
)

Для живой проверки https://3v4l.org/lJCse

Суреш Велусамы
источник
1
Обратите внимание, что для этого нужен как минимум PHP 5.6.0 (из-за использования константы ARRAY_FILTER_USE_KEY). В предыдущих версиях вы можете использовать: [code]
Netsurfer 01
20

С PHP 5.3 вы можете использовать эту preg_filterфункцию: здесь

$unprefixed_keys = preg_filter('/^foo-(.*)/', '$1', array_keys( $arr ));

// Result:
// $unprefixed_keys === array('bcd','def','xyz')
бизиклоп
источник
14
foreach($arr as $key => $value)
{
   if(preg_match('/^foo-/', $key))
   {
        // You can access $value or create a new array based off these values
   }
}
Тим Купер
источник
4
В этом случае регулярное выражение не требуется. Метод substr немного более эффективен
jfoucher
2
@jfoucher: Это правда, но лично читать легче.
Тим Купер,
3
Он может быть более гибким, но требует экранирования специальных символов preg, если вы не хотите выполнять сопоставление регулярного выражения. Так что используйте с осторожностью. Я бы посоветовал вам использовать RE только в том случае, если вам нужны RE.
Джейсон
7

Модификация функционального подхода Эриско ,

array_filter($signatureData[0]["foo-"], function($k) {
    return strpos($k, 'foo-abc') === 0;
}, ARRAY_FILTER_USE_KEY);

это сработало для меня.

Keyur
источник
Есть ли разница в производительности или полезности по сравнению с версией array_filter_key, предоставленной @erisco?
Жираф Джеффри
Нет. Думаю, разницы в производительности не должно быть. @MaciekSemik
Keyur
1
Это основная функция PHP, поэтому ей следует отдавать предпочтение.
Джереми Джон
2

В дополнение к ответу @Suresh Velusamy выше (для которого требуется как минимум PHP 5.6.0) вы можете использовать следующее, если используете более раннюю версию PHP:

<?php

$input = array(
    'abc' => 0,
    'foo-bcd' => 1,
    'foo-def' => 1,
    'foo-xyz' => 0,
);

$filtered = array_filter(array_keys($input), function($key) {
    return strpos($key, 'foo-') === 0;
});

print_r($filtered);

/* Output:
Array
(
    [1] => foo-bcd
    [2] => foo-def
    [3] => foo-xyz
)
// the numerical array keys are the position in the original array!
*/

// if you want your array newly numbered just add:
$filtered = array_values($filtered);

print_r($filtered);

/* Output:
Array
(
    [0] => foo-bcd
    [1] => foo-def
    [2] => foo-xyz
)
*/
Нетсёрфер
источник
1
Обратите внимание (или, по крайней мере, для меня), что это возвращает только совпадающий ключ массива. Ответ @ suresh-velusamy ниже извлекает фактический массив с исходной $key => $valueпарой. Об этом говорится в выводе этого ответа, но еще не упоминалось явно
Martie
0

Начиная с PHP5.6, ключи массива могут быть единственным объектом фильтрации с использованием константы / флага ARRAY_FILTER_USE_KEY .

Начиная с PHP7.4, стрелочные функции делают пользовательские функции более краткими и позволяют передавать значения в область пользовательской функции без use().

Начиная с PHP8, str_starts_with () может заменитьstrpos(...) === 0

Код: ( Демо )

$array = [
  'abc' => 0,
  'foo-bcd' => 1,
  'foo-def' => 1,
  'foo-xyz' => 0,
];

$prefix = 'foo';

var_export(
    array_filter(
        $array,
        fn($key) => str_starts_with($key, $prefix),
        ARRAY_FILTER_USE_KEY
    )
);

Вывод:

array (
  'foo-bcd' => 1,
  'foo-def' => 1,
  'foo-xyz' => 0,
)
Mickmackusa
источник