Сортировка массива объектов по полям объекта
Как я могу отсортировать этот массив объектов одним из своих полей, например name
или count
?
Array
(
[0] => stdClass Object
(
[ID] => 1
[name] => Mary Jane
[count] => 420
)
[1] => stdClass Object
(
[ID] => 2
[name] => Johnny
[count] => 234
)
[2] => stdClass Object
(
[ID] => 3
[name] => Kathy
[count] => 4354
)
....
Ответы
Ответ 1
Используйте usort, вот пример, адаптированный из руководства:
function cmp($a, $b) {
return strcmp($a->name, $b->name);
}
usort($your_data, "cmp");
Вы также можете использовать любой вызываемый в качестве второго аргумента. Вот несколько примеров:
Использование анонимных функций (из PHP 5.3)
usort($your_data, function($a, $b) {return strcmp($a->name, $b->name);});
Изнутри класса
usort($your_data, array($this, "cmp")); // "cmp" should be a method in the class
Использование функций стрелок (из PHP 7.4)
usort($your_data, fn($a, $b) => strcmp($a->name, $b->name));
Кроме того, если вы сравниваете числовые значения, fn($a, $b) => $a->count - $b->count
в качестве функции сравнения должна помочь.
Ответ 2
Лучше использовать закрытие
usort($your_data, function($a, $b)
{
return strcmp($a->name, $b->name);
});
Обратите внимание, что это не в документации PHP, но если вы используете закрытие 5.3+, поддерживаемое там, где могут быть предоставлены вызываемые аргументы.
Ответ 3
если вы используете php oop, вам может потребоваться изменить:
public static function cmp($a, $b)
{
return strcmp($a->name, $b->name);
}
//in this case FUNCTION_NAME would be cmp
usort($your_data, array('YOUR_CLASS_NAME','FUNCTION_NAME'));
Ответ 4
Если вы хотите отсортировать целочисленные значения:
// Desc sort
usort($array,function($first,$second){
return $first->number < $second->number;
});
// Asc sort
usort($array,function($first,$second){
return $first->number > $second->number;
});
ОБНОВЛЕНО со строкой, не забудьте преобразовать в тот же регистр (верхний или нижний)
// Desc sort
usort($array,function($first,$second){
return strtolower($first->text) < strtolower($second->text);
});
// Asc sort
usort($array,function($first,$second){
return strtolower($first->text) > strtolower($second->text);
});
Ответ 5
usort($array, 'my_sort_function');
var_dump($array);
function my_sort_function($a, $b)
{
return $a->name < $b->name;
}
Тот же код будет с полем count
.
Подробнее о usort
: http://ru2.php.net/usort
Btw, откуда вы взяли этот массив? Надеюсь, что не из базы данных?
Ответ 6
Вы можете использовать эту функцию (работает в версии PHP >= 5.3):
function sortArrayByKey(&$array,$key,$string = false,$asc = true){
if($string){
usort($array,function ($a, $b) use(&$key,&$asc)
{
if($asc) return strcmp(strtolower($a{$key}), strtolower($b{$key}));
else return strcmp(strtolower($b{$key}), strtolower($a{$key}));
});
}else{
usort($array,function ($a, $b) use(&$key,&$asc)
{
if($a[$key] == $b{$key}){return 0;}
if($asc) return ($a{$key} < $b{$key}) ? -1 : 1;
else return ($a{$key} > $b{$key}) ? -1 : 1;
});
}
}
Пример:
sortArrayByKey($yourArray,"name",true); //String sort (ascending order)
sortArrayByKey($yourArray,"name",true,false); //String sort (descending order)
sortArrayByKey($yourArray,"id"); //number sort (ascending order)
sortArrayByKey($yourArray,"count",false,false); //number sort (descending order)
Ответ 7
Вы можете использовать usort
, например:
usort($array,function($first,$second){
return strcmp($first->name, $second->name);
});
Ответ 8
Если все не удается, это другое решение:
$names = array();
foreach ($my_array as $my_object) {
$names[] = $my_object->name; //any object field
}
array_multisort($names, SORT_ASC, $my_array);
return $my_array;
Ответ 9
Нижняя сторона всех ответов здесь заключается в том, что они используют имена полей статические, поэтому я написал скорректированную версию в стиле ООП. Предположим, что вы используете методы getter, которые вы можете использовать непосредственно в этом классе, и используйте имя поля в качестве параметра. Возможно, кто-то сочтет это полезным.
class CustomSort{
public $field = '';
public function cmp($a, $b)
{
/**
* field for order is in a class variable $field
* using getter function with naming convention getVariable() we set first letter to uppercase
* we use variable variable names - $a->{'varName'} would directly access a field
*/
return strcmp($a->{'get'.ucfirst($this->field)}(), $b->{'get'.ucfirst($this->field)}());
}
public function sortObjectArrayByField($array, $field)
{
$this->field = $field;
usort($array, array("Your\Namespace\CustomSort", "cmp"));;
return $array;
}
}
Ответ 10
Простая альтернатива, позволяющая динамически определять поле, на котором основана сортировка:
$order_by = 'name';
usort($your_data, function ($a, $b) use ($order_by)
{
return strcmp($a->{$order_by}, $b->{$order_by});
});
Это основано на классе Closure, который позволяет анонимные функции. Он доступен с PHP 5.3.
Ответ 11
Если вам требуется локальное сравнение строк, вы можете использовать strcoll
вместо strcmp
.
Прежде чем использовать setlocale
с помощью LC_COLLATE
, чтобы установить локальную информацию, если необходимо.
usort($your_data,function($a,$b){
setlocale (LC_COLLATE, 'pl_PL.UTF-8'); // Example of Polish language collation
return strcoll($a->name,$b->name);
});
Ответ 12
Если вы используете это внутри Codeigniter, вы можете использовать следующие методы:
usort($jobs, array($this->job_model, "sortJobs")); // function inside Model
usort($jobs, array($this, "sortJobs")); // Written inside Controller.
@rmooney благодарим вас за предложение. Это действительно помогает мне.
Ответ 13
Спасибо за вдохновение, мне также пришлось добавить внешний параметр $переводчика
usort($listable_products, function($a, $b) {
global $translator;
return strcmp($a->getFullTitle($translator), $b->getFullTitle($translator));
});
Ответ 14
если вы хотите отсортировать даты
usort($threads,function($first,$second){
return strtotime($first->dateandtime) < strtotime($second->dateandtime);
});
Ответ 15
Если вам нужно отсортировать только по одному полю, то usort
- хороший выбор. Однако решение быстро становится беспорядочным, если вам нужно сортировать по нескольким полям. В этом случае можно использовать YaLinqo library *, которая реализует SQL-подобный синтаксис запросов для массивов и объектов. Он имеет довольно синтаксис для всех случаев:
$sortedByName = from($objects)->orderBy('$v->name');
$sortedByCount = from($objects)->orderBy('$v->count');
$sortedByCountAndName = from($objects)->orderBy('$v->count')->thenBy('$v->name');
Здесь '$v->count'
является сокращением для function ($v) { return $v->count; }
(либо можно использовать). Эти цепочки методов возвращают итераторы, но вы можете получить массивы, добавив ->toArray()
в конце, если вам это нужно.
*, разработанный мной
Ответ 16
Вы можете использовать sorted функцию Nspl:
use function \nspl\a\sorted;
use function \nspl\op\propertyGetter;
use function \nspl\op\methodCaller;
// Sort by property value
$sortedByCount = sorted($objects, propertyGetter('count'));
// Or sort by result of method call
$sortedByName = sorted($objects, methodCaller('getName'));
Ответ 17
Это то, что я имею для служебного класса
class Util
{
public static function sortArrayByName(&$arrayToSort, $meta) {
usort($arrayToSort, function($a, $b) use ($meta) {
return strcmp($a[$meta], $b[$meta]);
});
}
}
Назови это:
Util::sortArrayByName($array, "array_property_name");
Ответ 18
Вы можете использовать usort, как это
Если вы хотите отсортировать по номеру:
function cmp($a, $b)
{
if ($a == $b) {
return 0;
}
return ($a < $b) ? -1 : 1;
}
$a = array(3, 2, 5, 6, 1);
usort($a, "cmp");
Или Abc char:
function cmp($a, $b)
{
return strcmp($a["fruit"], $b["fruit"]);
}
$fruits[0]["fruit"] = "lemons";
$fruits[1]["fruit"] = "apples";
$fruits[2]["fruit"] = "grapes";
usort($fruits, "cmp");
Подробнее: https://www.php.net/manual/en/function.usort.php
Ответ 19
лучший простой способ
return array_values(array_sort($array, function ($value) {
return $value->key;
}));
Замените ключ на "ключ", который вы хотите заказать