Відповіді:
Ви або повторюєте масив, шукаючи конкретний запис (нормально одноразовий пошук) або будуєте хешмап за допомогою іншого асоціативного масиву.
Для перших щось подібне
$item = null;
foreach($array as $struct) {
if ($v == $struct->ID) {
$item = $struct;
break;
}
}
Дивіться це питання та наступні відповіді для отримання додаткової інформації щодо останнього - Довідковий масив PHP за кількома індексами
isset($item)
але я віддаю перевагу належним чином ініціалізувати змінні
if($v == $struct["ID"]){...
YurkamTim має рацію. Потрібна лише модифікація:
Після функції ($) вам потрібен вказівник на зовнішню змінну за допомогою "use (& $ searchValue)", і тоді ви можете отримати доступ до зовнішньої змінної. Також ви можете змінити його.
$neededObject = array_filter(
$arrayOfObjects,
function ($e) use (&$searchedValue) {
return $e->id == $searchedValue;
}
);
&
не потрібно при імпорті $searchedValue
в область закриття. &
Використовується для створення посилання , яка необхідна , тільки якщо $searchedValue
був змінений всередині затвора.
global
- це лише обмін даними у функціях! Але шкода, якщо це справді повільно. :(
$arr = [
[
'ID' => 1
]
];
echo array_search(1, array_column($arr, 'ID')); // prints 0 (!== false)
Я знайшов більш елегантне рішення тут . Адаптований до питання, це може виглядати так:
$neededObject = array_filter(
$arrayOfObjects,
function ($e) use ($searchedValue) {
return $e->id == $searchedValue;
}
);
array_filter
повертає масив і не зупиняється на першому знайденому значенні.
$searchedValue
всередині функції. Але зовні воно є.
$searchedValue
знаходиться поза межами області закриття. По-друге, як ви думаєте, як працюють ці масиви? Усі вони внутрішньо перебирають масив
$searchedValue
потрібно написатиfunction ($e) use ($searchedValue) {
Використання array_column для повторного індексування заощадить час, якщо вам потрібно знайти кілька разів:
$lookup = array_column($arr, NULL, 'id'); // re-index by 'id'
Тоді можна просто $lookup[$id]
за бажанням.
class ArrayUtils
{
public static function objArraySearch($array, $index, $value)
{
foreach($array as $arrayInf) {
if($arrayInf->{$index} == $value) {
return $arrayInf;
}
}
return null;
}
}
Використовувати його так, як ви хотіли, буде щось на зразок:
ArrayUtils::objArraySearch($array,'ID',$v);
Спробуйте
$entry = current(array_filter($array, function($e) use($v){ return $e->ID==$v; }));
робочий приклад тут
Виправляючи невелику помилку @YurkaTim , ваше рішення працює для мене, але додаєuse
:
Для використання $searchedValue
всередині функції одне рішення може бути use ($searchedValue)
після параметрів функціїfunction ($e) HERE
.
array_filter
функція повертає тільки на , $neededObject
якщо умова про повернення єtrue
Якщо $searchedValue
це рядок або ціле число:
$searchedValue = 123456; // Value to search.
$neededObject = array_filter(
$arrayOfObjects,
function ($e) use ($searchedValue) {
return $e->id == $searchedValue;
}
);
var_dump($neededObject); // To see the output
Якщо $searchedValue
є масив, де нам потрібно перевірити список:
$searchedValue = array( 1, 5 ); // Value to search.
$neededObject = array_filter(
$arrayOfObjects,
function ( $e ) use ( $searchedValue ) {
return in_array( $e->term_id, $searchedValue );
}
);
var_dump($neededObject); // To see the output
var_dump($neededObject);
:)
Мені іноді подобається використовувати функцію array_reduce () для здійснення пошуку. Він схожий на array_filter (), але не впливає на масив, що шукається, що дозволяє здійснювати кілька пошуків в одному масиві об'єктів.
$haystack = array($obj1, $obj2, ...); //some array of objects
$needle = 'looking for me?'; //the value of the object's property we want to find
//carry out the search
$search_results_array = array_reduce(
$haystack,
function($result_array, $current_item) use ($needle){
//Found the an object that meets criteria? Add it to the the result array
if ($current_item->someProperty == $needle){
$result_array[] = $current_item;
}
return $result_array;
},
array() //initially the array is empty (i.e.: item not found)
);
//report whether objects found
if (count($search_results_array) > 0){
echo "found object(s): ";
print_r($search_results_array[0]); //sample object found
} else {
echo "did not find object(s): ";
}
if ($current_item->someProperty == $needle){ $result_array[] = $current_item; }
Я зробив це за допомогою якихось ключових карт Java. Якщо ви це зробите, вам не потрібно щоразу перебирати масив об'єктів.
<?php
//This is your array with objects
$object1 = (object) array('id'=>123,'name'=>'Henk','age'=>65);
$object2 = (object) array('id'=>273,'name'=>'Koos','age'=>25);
$object3 = (object) array('id'=>685,'name'=>'Bram','age'=>75);
$firstArray = Array($object1,$object2);
var_dump($firstArray);
//create a new array
$secondArray = Array();
//loop over all objects
foreach($firstArray as $value){
//fill second key value
$secondArray[$value->id] = $value->name;
}
var_dump($secondArray);
echo $secondArray['123'];
вихід:
array (size=2)
0 =>
object(stdClass)[1]
public 'id' => int 123
public 'name' => string 'Henk' (length=4)
public 'age' => int 65
1 =>
object(stdClass)[2]
public 'id' => int 273
public 'name' => string 'Koos' (length=4)
public 'age' => int 25
array (size=2)
123 => string 'Henk' (length=4)
273 => string 'Koos' (length=4)
Henk
Я опублікував те, що використовую для ефективного вирішення цієї проблеми тут, використовуючи швидкий алгоритм двійкового пошуку: https://stackoverflow.com/a/52786742/1678210
Я не хотів копіювати ту саму відповідь. Хтось інший запитав це дещо інакше, але відповідь той самий.