it-swarm.com.ru

Сортировать массив объектов по полям объекта

Как я могу отсортировать этот массив объектов по одному из его полей, например, 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
        )

   ....
444
Alex

Используйте usort , вот пример, адаптированный из руководства:

function cmp($a, $b)
{
    return strcmp($a->name, $b->name);
}

usort($your_data, "cmp");

правки, импортированные из комментариев:

Если вы сортируете массив из класса, и ваша функция сортировки cmp также определена внутри класса, используйте это:

usort($your_data, array($this, "cmp"))
584
cambraca

Вот хороший способ использования замыканий

usort($your_data, function($a, $b)
{
    return strcmp($a->name, $b->name);
});

Обратите внимание, что этого нет в документации PHP, но если вы используете 5.3+, поддерживаются замыкания, где могут быть предоставлены вызываемые аргументы.

428
Scott Quinlan

если вы используете 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')); 
39
Doron Segal

Если вы хотите отсортировать целочисленные значения:

// 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);
});
31
Roman Yakoviv
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

Кстати, откуда вы взяли этот массив? Я надеюсь, что не из базы данных?

25
zerkms

Вы можете использовать эту функцию (работает в PHP Version> = 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)
8
PoengAlex

Вы можете использовать usort, например так:

usort($array,function($first,$second){
    return strcmp($first->name, $second->name);
});
3
Luca C.

Если все не получается, вот другое решение:

$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;
2
Adrian P.

Недостатком всех ответов здесь является то, что они используют имена полей static, поэтому я написал скорректированную версию в стиле OOP. Предполагая, что вы используете методы получения, вы можете напрямую использовать этот класс и использовать имя поля в качестве параметра. Возможно, кто-то найдет это полезным.

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;
    }
} 
2
Horst Jahns

Простая альтернатива, которая позволяет динамически определять поле, на котором основана сортировка:

$order_by = 'name';
usort($your_data, function ($a, $b) use ($order_by)
{
    return strcmp($a->{$order_by}, $b->{$order_by});
});

Это основано на Классе закрытия , который позволяет анонимные функции. Он доступен с PHP 5.3.

1
clami219

Спасибо за вдохновение, мне также пришлось добавить внешний параметр $ transator

usort($listable_products, function($a, $b) {
    global $translator;
    return strcmp($a->getFullTitle($translator), $b->getFullTitle($translator));
});
1
michalzuber

Если вам нужно локальное сравнение строк, вы можете использовать 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);
  });
1
Wilq

Если вы используете это внутри Codeigniter, вы можете использовать методы:

usort($jobs, array($this->job_model, "sortJobs"));  // function inside Model
usort($jobs, array($this, "sortJobs")); // Written inside Controller.

@rmooney спасибо за предложение. Это действительно помогает мне.

1
PHP Developer

Это то, что я имею для служебного класса

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");
0
Demodave

Если вам нужно отсортировать только по одному полю, тогда 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(), если вам это нужно.

* разработано мной

0
Athari

лучший простой способ

return array_values(array_sort($array, function ($value) {
  return $value->key;
}));

Замените ключ на «ключ», который вы хотите заказать

0
Diego Salguero

если вы хотите отсортировать даты

   usort($threads,function($first,$second){
        return strtotime($first->dateandtime) < strtotime($second->dateandtime);
    });
0
Nicolas Giszpenc

Вы можете использовать отсортировано функцию из 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'));
0
Ihor Burlachenko