Пошук багатовимірного масиву PHP за значенням


332

У мене є масив, де я хочу шукати uidі отримати ключ масиву.

Приклади

Припустимо, у нас є такий двовимірний масив:

$userdb = array(
    array(
        'uid' => '100',
        'name' => 'Sandra Shush',
        'pic_square' => 'urlof100'
    ),
    array(
        'uid' => '5465',
        'name' => 'Stefanie Mcmohn',
        'pic_square' => 'urlof100'
    ),
    array(
        'uid' => '40489',
        'name' => 'Michael',
        'pic_square' => 'urlof40489'
    )
);

Виклик функції search_by_uid(100)(uid першого користувача) повинен повернутися 0.

Виклик функції search_by_uid(40489)повинен повернутися 2.

Я спробував зробити петлі, але мені хочеться швидшого виконання коду.


що цікаво підкреслюють (і низький штрих) бібліотеки додають цю функцію до javascript ...
ErichBSchulz

9
Я написав сценарій для перевірки виконання декількох відповідей. Він генерує масив масивів 500k-членів і шукає через нього значення в останньому члені. Я порівняв таку функцію, як прийнята відповідь, з двома array_columnоднорядними відповідями. Я змінив їх усіх, щоб повернути фактично виявлений масив, а не лише ключ, тому що зазвичай це мій випадок використання. Метод функції набрав 0,361, середній показник пошуку 0,184 і клавіш-колонка 0,189 середньої мікро затримки понад 1000 запусків для кожного методу.
Джош

Відповіді:


474
function searchForId($id, $array) {
   foreach ($array as $key => $val) {
       if ($val['uid'] === $id) {
           return $key;
       }
   }
   return null;
}

Це спрацює. Ви повинні назвати це так:

$id = searchForId('100', $userdb);

Важливо знати, що якщо ви використовуєте ===оператори, типи порівняння повинні бути абсолютно однаковими, у цьому прикладі вам потрібно шукати stringабо просто використовувати ==замість цього ===.

На основі відповіді ангору . У пізніших версіях PHP ( >= 5.5.0) ви можете використовувати однолінійний.

$key = array_search('100', array_column($userdb, 'uid'));

Ось документація: http://php.net/manual/en/function.array-column.php .


8
Ви також можете це зробити без PHP 5.5 в одному вкладиші, використовуючи array_map замість array_column. Просто замініть array_column($userdb, 'uid')наarray_map(function($v){return $v['uid'];},$userdb)
Джессі Грін

1
Так, ви праві. Функції лямбда доступні з PHP 5.3. і краще array_search, чи не так?
Якуб Трунечек

@angoru Я думаю, що оригінальне рішення ( foreachцикл) буде працювати швидше, оскільки воно припиняється, як тільки знайдеться відповідність. Нове рішення має повторювати один раз весь масив для вилучення array_column, а потім перебирати його вдруге, щоб здійснити пошук (поки він не знайде збіг). Новіші рішення легше читати, більш стислі, але ОП спеціально викликала ефективність як питання
BeetleJuice

@ JakubTruneček. Я маю щось спільне з тим самим масивом, наведеним у питанні. Я хочу ім'я користувача з масиву, передавши id. Функція findUserName (40489) повинна повернути "Michael". Як це можливо?
Ашок Гуджар

@ JakubTruneček Привіт, я зіткнувся з цією проблемою у своєму коді, але у мене є зовсім інша річ. У моєму випадку значення uid присутнє кілька разів, тому мені потрібно отримати масив заснованого ключа.
Bhavin Thummar

314

Якщо ви використовуєте (PHP 5> = 5.5.0), вам не потрібно писати власну функцію для цього, просто напишіть цей рядок і це буде зроблено.

Якщо ви хочете лише один результат:

$key = array_search(40489, array_column($userdb, 'uid'));

Для отримання декількох результатів

$keys = array_keys(array_column($userdb, 'uid'), 40489);

Якщо у вас є асоціативний масив, як зазначено в коментарях, ви можете зробити це за допомогою:

$keys = array_keys(array_combine(array_keys($userdb), array_column($userdb, 'uid')),40489);

Якщо ви використовуєте PHP <5.5.0, ви можете скористатись цим спортом, спасибі ramsey!

Оновлення: я робив кілька простих орієнтирів, і форма декількох результатів здається найшвидшою, навіть швидшою, ніж спеціальна функція Jakub!


що робити, якщо значення, яке я шукаю (у цьому прикладі - 40489), з’являється більше, ніж один раз, і я хочу отримати всі ключі, які він з’являється?
Димитріс Папагеоргіу

якщо значення 40489 з'явиться більше, то один раз у масиві функція поверне масив ключів ... ?? @angoru
jishan

Якщо ви використовуєте другий регістр для отримання декількох результатів, ви отримуєте масив ключів.
ангору

2
Це не спрацювало для мене, коли ключ у $ userdb не запускався як 0,1, 2 і т. Д. І скажімо, що ключ - 1234,4566 тощо. Отримані ключі після масиву_визначення завжди становлять 0,1,2 і так on
Кауштув

1
Це не працюватиме з асоціативним масивом, однак ви можете його обійти так: array_search(40489, array_combine(array_keys($userdb), array_column($userdb, 'uid')))
Джон Меллор

32

У пізніших версіях PHP (> = 5.5.0) ви можете використовувати цей однолінійний:

$key = array_search('100', array_column($userdb, 'uid'));

Просто поставте результат array_column у певну змінну, уникаючи, щоб array_column викликався для кожного результату в масиві.
Майконн

26

Створюючи чудову відповідь Якуба , ось більш узагальнений пошук, який дозволить вказати ключ (не лише для uid):

function searcharray($value, $key, $array) {
   foreach ($array as $k => $val) {
       if ($val[$key] == $value) {
           return $k;
       }
   }
   return null;
}

Використання: $results = searcharray('searchvalue', searchkey, $array);


Це дуже корисно, я відчуваю, що я зараз вирішую свою проблему за допомогою цього рішення, але у мене все ще виникає проблема. Не могли б ви дати зрозуміти? Питання можна знайти тут: stackoverflow.com/questions/28704644 / ...
jasenmp

19

Я знаю, що на це вже відповіли, але я скористався цим і трохи більше розширив його у своєму коді, щоб у вас не було пошуку лише uid. Я просто хочу поділитися цим для всіх, кому ця функція може знадобитися.

Ось мій приклад, і, майте на увазі, це моя перша відповідь. Я вийняв масив param, тому що мені потрібно було шукати лише один конкретний масив, але ви могли легко додати його. Я хотів по суті шукати більше, ніж просто uid.

Також у моїй ситуації може бути кілька клавіш для повернення в результаті пошуку за іншими полями, які можуть бути не унікальними.

 /**
     * @param array multidimensional 
     * @param string value to search for, ie a specific field name like name_first
     * @param string associative key to find it in, ie field_name
     * 
     * @return array keys.
     */
     function search_revisions($dataArray, $search_value, $key_to_search) {
        // This function will search the revisions for a certain value
        // related to the associative key you are looking for.
        $keys = array();
        foreach ($dataArray as $key => $cur_value) {
            if ($cur_value[$key_to_search] == $search_value) {
                $keys[] = $key;
            }
        }
        return $keys;
    }

Пізніше я закінчив писати це, щоб дозволити мені шукати інше значення та асоціативний ключ. Тож мій перший приклад дозволяє шукати значення в будь-якому конкретному асоціативному ключі та повертати всі збіги.

Цей другий приклад показує, де в певному асоціативному ключі (first_name) знайдено значення ('Taylor') І інше значення (true) в іншому асоціативному ключі (найнято), і повертає всі збіги (Ключі, де люди з прізвищем "Тейлор" І працюють).

/**
 * @param array multidimensional 
 * @param string $search_value The value to search for, ie a specific 'Taylor'
 * @param string $key_to_search The associative key to find it in, ie first_name
 * @param string $other_matching_key The associative key to find in the matches for employed
 * @param string $other_matching_value The value to find in that matching associative key, ie true
 * 
 * @return array keys, ie all the people with the first name 'Taylor' that are employed.
 */
 function search_revisions($dataArray, $search_value, $key_to_search, $other_matching_value = null, $other_matching_key = null) {
    // This function will search the revisions for a certain value
    // related to the associative key you are looking for.
    $keys = array();
    foreach ($dataArray as $key => $cur_value) {
        if ($cur_value[$key_to_search] == $search_value) {
            if (isset($other_matching_key) && isset($other_matching_value)) {
                if ($cur_value[$other_matching_key] == $other_matching_value) {
                    $keys[] = $key;
                }
            } else {
                // I must keep in mind that some searches may have multiple
                // matches and others would not, so leave it open with no continues.
                $keys[] = $key;
            }
        }
    }
    return $keys;
}

Використання функції

$data = array(
    array(
        'cust_group' => 6,
        'price' => 13.21,
        'price_qty' => 5
    ),
    array(
        'cust_group' => 8,
        'price' => 15.25,
        'price_qty' => 4
    ),
    array(
        'cust_group' => 8,
        'price' => 12.75,
        'price_qty' => 10
    )
);

$findKey = search_revisions($data,'8', 'cust_group', '10', 'price_qty');
print_r($findKey);

Результат

Array ( [0] => 2 ) 

10

Здається, array_filter буде підходящим рішенням для цього ...

$userdb=Array
(
    (0) => Array
        (
            (uid) => '100',
            (name) => 'Sandra Shush',
            (url) => 'urlof100'
        ),

    (1) => Array
        (
            (uid) => '5465',
            (name) => 'Stefanie Mcmohn',
            (pic_square) => 'urlof100'
        ),

    (2) => Array
        (
            (uid) => '40489',
            (name) => 'Michael',
            (pic_square) => 'urlof40489'
        )
);

PHP-код

<?php 
$search = 5465;
$found = array_filter($userdb,function($v,$k) use ($search){
  return $v['uid'] == $search;
},ARRAY_FILTER_USE_BOTH) // With latest PHP third parameter is mandatory.. Available Values:- ARRAY_FILTER_USE_BOTH OR ARRAY_FILTER_USE_KEY  

$values= print_r(array_value($found)); 
$keys =  print_r(array_keys($found)); 

@BEJAM SHIVA PRASAD, чи можете ви мені допомогти з цим stackoverflow.com/questions/44721195/… ?
Валай

показує помилку:syntax error, unexpected '=>' (T_DOUBLE_ARROW), expecting ';'
Шихас

Чи можете ви, будь ласка, вставити більше інформації, я маю на увазі, який рядок та ваш код та структуру масиву.
БЕЖГАМ ШІВА ПРАСАД

@Shihas Я оновив відповідь, впевнений, що це буде вирішено.
BEJGAM SHIVA PRASAD

9

Я змінив один із наведених нижче прикладів функції опису array_search . Функція searchItemsByKeyповертає всі значення (-ла) ключем $ з багатовимірного масиву (N рівнів). Можливо, комусь це було б корисно. Приклад:

 $arr = array(
     'XXX'=>array(
               'YYY'=> array(
                    'AAA'=> array(
                          'keyN' =>'value1'
                   )
               ),
              'ZZZ'=> array(
                    'BBB'=> array(
                          'keyN' => 'value2'
                   )
               )
              //.....
           )
);


$result = searchItemsByKey($arr,'keyN');

print '<pre>';
print_r($result);
print '<pre>';
// OUTPUT
Array
(
  [0] => value1
  [1] => value2
)

Код функції:

function searchItemsByKey($array, $key)
{
   $results = array();

  if (is_array($array))
  {
    if (isset($array[$key]) && key($array)==$key)
        $results[] = $array[$key];

    foreach ($array as $sub_array)
        $results = array_merge($results, searchItemsByKey($sub_array, $key));
  }

 return  $results;
}

7

Ось один вкладиш для того ж,

$pic_square = $userdb[array_search($uid,array_column($userdb, 'uid'))]['pic_square'];

3

Хоча це старе запитання і має прийняту відповідь, Думаю, я б запропонував одну зміну прийнятої відповіді. Отже, спочатку я погоджуюсь, що прийнята відповідь тут правильна.

function searchArrayKeyVal($sKey, $id, $array) {
   foreach ($array as $key => $val) {
       if ($val[$sKey] == $id) {
           return $key;
       }
   }
   return false;
}

Заміна замість заданого 'uid' параметром у функції, тому тепер виклик коду нижче означає, що ви можете використовувати одну функцію для кількох типів масиву. Невелика зміна, але одна, яка робить незначну зміну.

    // Array Data Of Users
$userdb = array (
    array ('uid' => '100','name' => 'Sandra Shush','url' => 'urlof100' ),
    array ('uid' => '5465','name' => 'Stefanie Mcmohn','url' => 'urlof100' ),
    array ('uid' => '40489','name' => 'Michael','url' => 'urlof40489' ),
);

// Obtain The Key Of The Array
$arrayKey = searchArrayKeyVal("uid", '100', $userdb);
if ($arrayKey!==false) {
    echo "Search Result: ", $userdb[$arrayKey]['name'];
} else {
    echo "Search Result can not be found";
}

Приклад PHiddle Fiddle


Ще одна причина, по якій я використовую SOF ... простіше в Google і знайти власний код або щось, що я пам’ятаю .. +1 для мого власного публічного сховища
Angry 84

До речі, ви встановлюєте результат, nullа потім у коді ви порівнюєте його false.
Таха Паксу

Виправлено поверненням false, але нуль може бути кращим у випадку перевірки булевих
Angry 84

3

Я хочу перевірити tha в наступному масиві, $arrчи є 'abc' в підмасивах чи ні

$arr = array(
    array(
        'title' => 'abc'
    )
);

Тоді я можу цим скористатися

$res = array_search('abc', array_column($arr, 'title'));
if($res == ''){
    echo 'exists';
} else {
    echo 'notExists';
}

Я думаю, що це найпростіший спосіб визначення


1

Мені довелося використовувати функцію un, яка знаходить всі елементи в масиві. Тож я змінив функцію, яку виконував Якуб Трунечек, наступним чином:

function search_in_array_r($needle, $array) {
    $found = array();
    foreach ($array as $key => $val) {
        if ($val[1] == $needle) {
            array_push($found, $val[1]);
        }
    }
    if (count($found) != 0)
        return $found;
    else
        return null;
}

1
/**
 * searches a simple as well as multi dimension array
 * @param type $needle
 * @param type $haystack
 * @return boolean
 */
public static function in_array_multi($needle, $haystack){
    $needle = trim($needle);
    if(!is_array($haystack))
        return False;

    foreach($haystack as $key=>$value){
        if(is_array($value)){
            if(self::in_array_multi($needle, $value))
                return True;
            else
               self::in_array_multi($needle, $value);
        }
        else
        if(trim($value) === trim($needle)){//visibility fix//
            error_log("$value === $needle setting visibility to 1 hidden");
            return True;
        }
    }

    return False;
}

1

ви можете використовувати цю функцію; https://github.com/serhatozles/ArrayAdvancedSearch

<?php 
include('ArraySearch.php');

$query = "a='Example World' and b>='2'";

$Array = array(
'a' => array('d' => '2'),
array('a' => 'Example World','b' => '2'),
array('c' => '3'), array('d' => '4'),
);

$Result = ArraySearch($Array,$query,1);

echo '<pre>';
print_r($Result);
echo '</pre>'; 

// Output:
// Array
// (
//    [0] => Array
//        (
//            [a] => Example World
//            [b] => 2
//        )
//
// )

1
$a = ['x' => ['eee', 'ccc'], 'b' => ['zzz']];

$found = null;
$search = 'eee';

array_walk($a, function ($k, $v) use ($search, &$found) {
    if (in_array($search, $k)) {
        $found = $v;
    }
});

var_dump($found);

1

Спробуйте це

<?php
 function recursive_array_search($needle,$haystack) {
    foreach($haystack as $key=>$value) {
        $current_key=$key;
        if($needle===$value OR (is_array($value) && 
            recursive_array_search($needle,$value) !== false)) {
             return $current_key;
        }
    }
    return false;
 }
 ?>

1

Просто поділіться, може, це сподобається.

if( ! function_exists('arraySearchMulti')){
function arraySearchMulti($search,$key,$array,$returnKey=false)
{
    foreach ($array as $k => $val) {
        if (isset($val[$key])) {
            if ((string)$val[$key] == (string)$search) {
                return ($returnKey ? $k : $val);
            }
        }else{
            return (is_array($val) ? arraySearchMulti($search,$key,$val,$returnKey) : null);
        }
    }
    return null;
}}

0

Спробуйте і це

function search_in_array($srchvalue, $array)
{
    if (is_array($array) && count($array) > 0)
    {
        $foundkey = array_search($srchvalue, $array);
        if ($foundkey === FALSE)
        {
            foreach ($array as $key => $value)
            {
                if (is_array($value) && count($value) > 0)
                {
                    $foundkey = search_in_array($srchvalue, $value);
                    if ($foundkey != FALSE)
                        return $foundkey;
                }
            }
        }
        else
            return $foundkey;
    }
}

0
for( $i =0; $i < sizeof($allUsers); $i++)
    {   
    $NEEDLE1='firstname';
    $NEEDLE2='emailAddress';
    $sterm='Tofind';
     if(isset($allUsers[$i][$NEEDLE1]) && isset($allUsers[$i][$NEEDLE2])
        {
            $Fname= $allUsers[$i][$NEEDLE1];
            $Lname= $allUsers[$i][$NEEDLE2];

            $pos1 = stripos($Fname, $sterm);
            $pos2=stripos($Lname, $sterm);//not case sensitive 

            if($pos1 !== false ||$pos2 !== false)
            {$resultsMatched[] =$allUsers[$i];}
            else
            {   continue;}              
        }

}
Print_r($resultsMatched); //will give array for matched values even partially matched

За допомогою вищевказаного коду можна знайти будь-які (частково збігаються) дані з будь-якого стовпця в 2D масиві, щоб ідентифікатор користувача можна було знайти так, як потрібно.


Будь ласка, додайте фразу, щоб пояснити, чому це відповідає на питання
Лоренц Мейєр

з допомогою наведеного вище коду можна знайти будь-які (частково збігаються) дані з будь-якого стовпця в 2D масиві, щоб ідентифікатор користувача можна було знайти, як потрібно в питанні
sandeep sharma

0

Розширюючись на створену функцію @mayhem, цей приклад був би більш "нечітким" пошуком у випадку, якщо ви просто хочете відповідати частині ( більшості ) рядка пошуку:

 function searchArrayKeyVal($sKey, $id, $array) {
    foreach ($array as $key => $val) {
        if (strpos(strtolower($val[$sKey]), strtolower(trim($id))) !== false) {
            return $key;
        }
    }
         return false;
 }

Наприклад, значення в масиві вітається в Нью-Йорку! і ви хотіли першого екземпляра просто "Нью-Йорка!"


0
$search1 = 'demo';
$search2 = 'bob';
$arr = array('0' => 'hello','1' => 'test','2' => 'john','3' => array('0' => 'martin', '1' => 'bob'),'4' => 'demo');
foreach ($arr as $value) { 
    if (is_array($value)) { 
        if (in_array($search2, $value)) { 
            echo "successsfully";    
            //execute your code 
        }
    } else {  
        if ($value == $search1) { 
            echo "success";
        }
    }
 }

0

Якщо питання, тобто

$a = [
     [
       "_id" => "5a96933414d48831a41901f2",
       "discount_amount" => 3.29,
       "discount_id" => "5a92656a14d488570c2c44a2",
     ],
     [
       "_id" => "5a9790fd14d48879cf16a9e8",
       "discount_amount" => 4.53,
       "discount_id" => "5a9265b914d488548513b122",
     ],
     [
       "_id" => "5a98083614d488191304b6c3",
       "discount_amount" => 15.24,
       "discount_id" => "5a92806a14d48858ff5c2ec3",
     ],
     [
       "_id" => "5a982a4914d48824721eafe3",
       "discount_amount" => 45.74,
       "discount_id" => "5a928ce414d488609e73b443",
     ],
    [
       "_id" => "5a982a4914d48824721eafe55",
       "discount_amount" => 10.26,
       "discount_id" => "5a928ce414d488609e73b443",
     ],
   ];

Відповідь:

function searchForId($id, $array) {
    $did=0;
    $dia=0;
   foreach ($array as $key => $val) {
       if ($val['discount_id'] === $id) {
           $dia +=$val['discount_amount'];
           $did++;
       }
   }
    if($dia != '') {
        echo $dia;
        var_dump($did);
    }
   return null;
};
print_r(searchForId('5a928ce414d488609e73b443',$a));

0

моє рішення:

function searchArrayForField($array, $field, $value) {
    $i = 0;
    foreach ($array as &$row) {
        if ($row[$field] === $value) {
            return $i;
        }
        $i++
    }
    return '';
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.