Як сортувати багатовимірний масив за значенням?


1129

Як можна сортувати цей масив за значенням ключа "замовлення"? Незважаючи на те, що значення наразі є послідовними, вони не завжди будуть.

Array
(
    [0] => Array
        (
            [hashtag] => a7e87329b5eab8578f4f1098a152d6f4
            [title] => Flower
            [order] => 3
        )

    [1] => Array
        (
            [hashtag] => b24ce0cd392a5b0b8dedc66c25213594
            [title] => Free
            [order] => 2
        )

    [2] => Array
        (
            [hashtag] => e7d31fc0602fb2ede144d18cdffd816b
            [title] => Ready
            [order] => 1
        )
)


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

Відповіді:


1732

Спробуйте usort. Якщо ви все ще користуєтесь PHP 5.2 або новішою версією, вам потрібно спочатку визначити функцію сортування:

function sortByOrder($a, $b) {
    return $a['order'] - $b['order'];
}

usort($myArray, 'sortByOrder');

Починаючи з PHP 5.3, ви можете використовувати анонімну функцію:

usort($myArray, function($a, $b) {
    return $a['order'] - $b['order'];
});

І, нарешті, з PHP 7 ви можете використовувати оператор космічного корабля :

usort($myArray, function($a, $b) {
    return $a['order'] <=> $b['order'];
});

Щоб поширити це на багатовимірне сортування, посилайтеся на другий / третій елементи сортування, якщо перший дорівнює нулю - найкраще пояснено нижче. Ви також можете використовувати це для сортування по піделементах.

usort($myArray, function($a, $b) {
    $retval = $a['order'] <=> $b['order'];
    if ($retval == 0) {
        $retval = $a['suborder'] <=> $b['suborder'];
        if ($retval == 0) {
            $retval = $a['details']['subsuborder'] <=> $b['details']['subsuborder'];
        }
    }
    return $retval;
});

Якщо вам потрібно зберегти ключові асоціації, використовуйте uasort()- див. Порівняння функцій сортування масивів у посібнику


2
У мене фразування трохи відключено, я її відредагую. Я мав на увазі те, що якщо ви не на PHP 5.3, вам потрібно створити спеціальну функцію саме для цього конкретного сортування (що якось не дуже елегантно). В іншому випадку ви можете використовувати анонімну функцію саме там.
Християнський Студер

23
@Jonathan: Більшість роботи, яку робить PHP, ти не можеш реально побачити. Він бере масив і починається з двох елементів, які він передав цій функції, визначеній користувачем. Тоді ваша функція відповідає за їх порівняння: Якщо перший елемент більший за другий, поверніть додатне ціле число, якщо воно менше, поверніть негативне. Якщо вони рівні, поверніть 0. PHP потім посилає два інші елементи до вашої функції і продовжує робити це, поки масив не буде відсортований. Функція тут дуже коротка, вона може бути набагато складнішою, якби ви не порівнювали цілі числа.
Християнський Студент

61
Підказка: використовуйте, uasort()якщо потрібно зберегти клавіші масиву.
thaddeusmt

15
Будьте уважні, якщо значення сортування є десятковими числами. Якщо функція сортування отримує $ a = 1 і $ b = 0,1, різниця дорівнює 0,9, але функція повертає int, в цьому випадку 0, тому $ a і $ b вважаються рівними і сортуються неправильно. Більш надійно порівняти, якщо $ a більше, ніж $ b або дорівнює, і повернути відповідно -1, 1 або 0.
Ґренланді

37
Мені знадобилося деякий час, щоб дізнатися. Для сортування зворотного порядку (DESC) ви можете переключити $ a і $ b. Так $ b ['замовлення'] - $ a ['замовлення']
JanWillem

285
function aasort (&$array, $key) {
    $sorter=array();
    $ret=array();
    reset($array);
    foreach ($array as $ii => $va) {
        $sorter[$ii]=$va[$key];
    }
    asort($sorter);
    foreach ($sorter as $ii => $va) {
        $ret[$ii]=$array[$ii];
    }
    $array=$ret;
}

aasort($your_array,"order");

7
@ noc2spam Я радий допомогти, але врахуйте наступні пропозиції студента, які, ймовірно, більш ефективні та, безумовно, більш охайні!
o0 '.

1
@Lohoris впевнений чувак, я це теж перевіряю. Вчора був би жорсткий день в офісі, якби я не знайшов цього питання :-)
Гоголь

Хм не можу додати відповідь .. ну я поклав це сюди, мені не потрібні ці дурні точки: так що для багатовимірного сортування його (майже) те ж саме (srry ви повинні скопіювати вставити і переформатувати): function aasort (& $ array , $ key1, $ key2, $ key3) {$ sorter = array (); $ ret = масив (); скидання ($ масив); foreach ($ масив як $ ii => $ va) {$ sorter [$ ii] = getPrice ($ va [$ key1] [$ key2] [$ key3]); } arsort ($ сортування); foreach ($ sorter як $ ii => $ va) {$ ret [$ ii] = $ масив [$ ii]; } $ масив = $ ret; }
Jaxx0rr

3
Набагато простіше застосувати, ніж відповідь вище
Марсель

1
ВИ ЧУДОВІ!! РОБОТА ПОДОБАЄТЬСЯ ЧАРМУ ДЛЯ PHP 7.2
Kamlesh

270

Я використовую цю функцію:

function array_sort_by_column(&$arr, $col, $dir = SORT_ASC) {
    $sort_col = array();
    foreach ($arr as $key=> $row) {
        $sort_col[$key] = $row[$col];
    }

    array_multisort($sort_col, $dir, $arr);
}


array_sort_by_column($array, 'order');

3
Працює дуже добре. Унікальне рішення, здатне додати напрямок сортування. Дякую!
Іво Перейра

2
Для альтернативи, яка підтримує впорядкування впорядкування та безліч додаткових функцій, ви можете поглянути на мою відповідь тут - вона також має перевагу в тому, що вона не використовується, array_multisortі тому не потрібно попередньо розміщувати $sort_col, економлячи час та пам'ять. .
Джон

Для мене чудово працює, але ... чому потрібно вказувати &$arrзамість просто $arr?
Раду Мурзеа

2
@RaduMurzea, тому масив передається посиланням і може бути змінений функцією, а не функцією, що отримує копію
Том Хей

1
@AdrianP. : масив передається за посиланням, тому він буде змінювати пройдений масив, а не повертати відсортовану копію
Том Хей

72

Зазвичай я використовую usort і передаю власну функцію порівняння. У цьому випадку це дуже просто:

function compareOrder($a, $b)
{
  return $a['order'] - $b['order'];
}
usort($array, 'compareOrder');

У PHP 7 використовується оператор космічного корабля:

usort($array, function($a, $b) {
    return $a['order'] <=> $b['order'];
});

4
Чорт, я був на 30 секунд повільніше. Хіба це не $ a - $ b?
Christian Studer

3
Я завжди розумію це неправильно. Дозвольте подумати з посібника: повернене значення повинно бути менше нуля, якщо перший аргумент вважається меншим за другий. Тож якщо $a['order']це 3, а $b['order']це 6, я поверну -3. Правильно?
Ян Фабрі

3
Ну, ви повернете b - a, значить, це буде 3. І таким чином відсортовано неправильно.
Християнський Студер

26
Ага. ДОБРЕ. Я використовував не логічну арифметику, де ідея у вашій голові не відповідає словам, які ви створюєте. Це вивчається найчастіше в п'ятницю.
Ян Фабрі

в деяких випадках вище відповідь є повертається результат неправильно .. посилання stackoverflow.com/questions/50636981 / ...
Білал Ахмед

45

Один із підходів для досягнення цього був би таким

    $new = [
              [
                'hashtag' => 'a7e87329b5eab8578f4f1098a152d6f4',
                'title' => 'Flower',
                'order' => 3,
              ],

              [
                'hashtag' => 'b24ce0cd392a5b0b8dedc66c25213594',
                'title' => 'Free',
                'order' => 2,
              ],

              [
                'hashtag' => 'e7d31fc0602fb2ede144d18cdffd816b',
                'title' => 'Ready',
                'order' => 1,
              ],
    ];

    $keys = array_column($new, 'order');

    array_multisort($keys, SORT_ASC, $new);

    var_dump($new);

Результат:

    Array
    (
        [0] => Array
            (
                [hashtag] => e7d31fc0602fb2ede144d18cdffd816b
                [title] => Ready
                [order] => 1
            )

        [1] => Array
            (
                [hashtag] => b24ce0cd392a5b0b8dedc66c25213594
                [title] => Free
                [order] => 2
            )

        [2] => Array
            (
                [hashtag] => a7e87329b5eab8578f4f1098a152d6f4
                [title] => Flower
                [order] => 3
            )

    )

18

Для сортування масиву за значенням ключа "title" використовуйте:

uasort($myArray, function($a, $b) {
    return strcmp($a['title'], $b['title']);
});

strcmp порівняйте рядки.

uasort () підтримує ключі масиву, як вони були визначені.


strcmp - помітна допомога для типів varchar, мені сподобалась ця відповідь коротко і швидко
StudioX

17
$sort = array();
$array_lowercase = array_map('strtolower', $array_to_be_sorted);
array_multisort($array_lowercase, SORT_ASC, SORT_STRING, $alphabetically_ordered_array);

Це стосується як великих, так і малих букв.


4
Як це відповідає на питання сортування за певним ключем масиву?
Сеано

12

Використовувати array_multisort(),array_map()

array_multisort(array_map(function($element) {
      return $element['order'];
  }, $array), SORT_ASC, $array);

print_r($array);

DEMO


Зауважте, що такий підхід надсилатиме повідомлення. array_multisortпосилання - це перший параметр.
Камі Ян

5

Найбільш гнучким підходом було б використання цього методу

Arr::sortByKeys(array $array, $keys, bool $assoc = true): array

ось чому:

  • Ви можете сортувати за будь-яким ключем (також вкладеним як 'key1.key2.key3'або ['k1', 'k2', 'k3'])

  • Працює як на асоціативних, так і не асоціативних масивах ( $assocпрапор)

  • Він не використовує посилання - повертає новий відсортований масив

У вашому випадку це було б так просто, як:

$sortedArray = Arr::sortByKeys($array, 'order');

Цей метод є частиною цієї бібліотеки .


1

Давайте поглянемо на це: php НЕ має просту функцію поза вікном, щоб правильно обробляти кожен сценарій сортування масиву.

Ця програма є інтуїтивно зрозумілою, що означає швидку налагодження та обслуговування:

// automatic population of array
$tempArray = array();
$annotations = array();
// ... some code
// SQL $sql retrieves result array $result 
// $row[0] is the ID, but is populated out of order (comes from 
// multiple selects populating various dimensions for the same DATE 
// for example
while($row = mysql_fetch_array($result)) {
    $needle = $row[0];
    arrayIndexes($needle);  // create a parallel array with IDs only
    $annotations[$needle]['someDimension'] = $row[1]; // whatever
}
asort($tempArray);
foreach ($tempArray as $arrayKey) {
    $dataInOrder = $annotations[$arrayKey]['someDimension']; 
    // .... more code
}

function arrayIndexes ($needle) {
    global $tempArray;
    if (!in_array($needle,$tempArray)) {
        array_push($tempArray,$needle);
    }
}

4
"Давайте визначимося: php НЕ має просту функцію поза коробкою, щоб правильно обробляти кожен сценарій сортування масиву." Саме так призначені usort / ksort / asort для ^^ '
Бен Кассінат

3
Насправді PHP має багато функцій сортування, які можна використовувати для обробки кожного сценарію сортування масиву.
аксіак

Що стосується налагодження та обслуговування, використання globalвеличезного червоного прапора і, як правило, не рекомендується . Чому mysql_fetch_arrayдемонструється це питання замість вихідного масиву ОП, і що немає пояснень того, що робить ваш код, і чого можна очікувати результату? Загалом це дуже складний підхід для досягнення бажаного кінцевого результату.
fyrye

@tonygil Я не в змозі визначити, які очікувані результати ви отримаєте з вашої відповіді та набору даних ОП. Вам це може бути очевидно, але я не знаю, як ваша відповідь відповідає на питання ОП. Однак ви можете пройти посиланням замість використання globalдив.: 3v4l.org/FEeFC Це створює чітко визначену змінну, а не ту, яку можна змінити та отримати доступ до неї в усьому світі.
fyrye

0

Ви також можете сортувати багатовимірний масив для кількох значень, таких як

$arr = [
    [
        "name"=> "Sally",
        "nick_name"=> "sal",
        "availability"=> "0",
        "is_fav"=> "0"
    ],
    [
        "name"=> "David",
        "nick_name"=> "dav07",
        "availability"=> "0",
        "is_fav"=> "1"
    ],
    [
        "name"=> "Zen",
        "nick_name"=> "zen",
        "availability"=> "1",
        "is_fav"=> "0"
    ],
    [
        "name"=> "Jackson",
        "nick_name"=> "jack",
        "availability"=> "1",
        "is_fav"=> "1"
    ],
    [
        "name"=> "Rohit",
        "nick_name"=> "rod",
        "availability"=> "0",
        "is_fav"=> "0"
    ],

];

з

usort($arr,function($a,$b){
    $c = $b['is_fav'] - $a['is_fav'];
    $c .= $b['availability'] - $a['availability'];
    $c .= strcmp($a['nick_name'],$b['nick_name']);
    return $c;
});

Вихід із використанням print_r($arr):

Array
(
    [0] => Array
        (
            [name] => Jackson
            [nick_name] => jack
            [availability] => 1
            [is_fav] => 1
        )

    [1] => Array
        (
            [name] => David
            [nick_name] => dav07
            [availability] => 0
            [is_fav] => 1
        )

    [2] => Array
        (
            [name] => Zen
            [nick_name] => zen
            [availability] => 1
            [is_fav] => 0
        )

    [3] => Array
        (
            [name] => Rohit
            [nick_name] => rod
            [availability] => 0
            [is_fav] => 0
        )

    [4] => Array
        (
            [name] => Sally
            [nick_name] => sal
            [availability] => 0
            [is_fav] => 0
        )

)

PS) використання strcmp було б хорошим варіантом для порівняння рядків.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.