web-dev-qa-db-ja.com

PHP 2つのフィールド値で配列をソート

私はこのような配列を持っています

 Array(
 [0] => Array( "destination" => "Sydney"、
 "airlines" => "airline_1"、
 "one_way_fare" => 100、
 "return_fare => 300 
)、
 [2] => Array(" destination "=>" Sydney "、
" airlines " => "airline_2"、
 "one_way_fare" => 150、
 "return_fare => 350 
)、
 [3] => Array(" destination " => "Sydney"、
 "airlines" => "airline_3"、
 "one_way_fare" => 180、
 "return_fare => 380 
)
)

Return_fare asc、one_way_fare ascで値をソートするにはどうすればよいですか?

array_multisort() を試しましたが、結局データが混ざってしまいました。

asortは1次元配列でのみ機能します。2つ以上の値で並べ替える必要があります。SQLのように、field1 asc、field2 ascのように並べ替えるにはどうすればよいですか。

61
flyclassic

array_multisort() は正しい関数です。何らかの方法で混乱しているに違いありません。

_// Obtain a list of columns
foreach ($data as $key => $row) {
    $return_fare[$key]  = $row['return_fare'];
    $one_way_fare[$key] = $row['one_way_fare'];
}

// Sort the data with volume descending, edition ascending
array_multisort($return_fare, SORT_ASC, $one_way_fare, SORT_ASC, $data);
_

PHPのarray_multisort()のマニュアルページのコメントを見ると、非常に役立つarray_orderby()関数を見つけることができます。

_$sorted = array_orderby($data, 'return_fare', SORT_ASC, 'one_way_fare', SORT_ASC);
_

ループを回避するには、array_column()を使用します(PHP 5.5.0)以降):

_array_multisort(array_column($data, 'return_fare'),  SORT_ASC,
                array_column($data, 'one_way_fare'), SORT_ASC,
                $data);
_
97
Tatu Ulmanen

最初に列配列を構築する必要があるarray_multisort()に加えて、 usort() もあり、これはそのようなことを必要としません。

usort($data, function($a, $b) { 
    $rdiff = $a['return_fare'] - $b['return_fare'];
    if ($rdiff) return $rdiff; 
    return $a['one_way_fare'] - $b['one_way_fare']; 
}); // anonymous function requires PHP 5.3 - use "normal" function earlier
34
etarion

または、次のようにuasortを使用できます

uasort($arr, function($a,$b){
    $c = $a['return_fare'] - $b['return_fare'];
    $c .= $a['one_way_fare'] - $b['one_way_fare'];
    return $c;
});

Fiddle

宇宙船オペレーターを使用した別の例。

usort($data, function($a, $b) { 
    return $a['return_fare'] <=> $b['return_fare'] ?: $a['one_way_fare'] <=> $b['one_way_fare'] 
});
5
user634545

並べ替えたいアイテムの数に関係なく、generalizedになる方法でこれに答えます!

return_fare、次にone_way_fareでソート:

usort($data, function($a, $b) {
   if ($a['return_fare'] != $b['return_fare']) {
      return $a['return_fare'] <=> $b['return_fare'];
   }

   return $a['one_way_fare'] <=> $b['one_way_fare'];
});

return_fare、_one_way_fare、_destinationの順に並べ替えます:

usort($data, function($a, $b) {
   if ($a['return_fare'] != $b['return_fare']) {
      return $a['return_fare'] <=> $b['return_fare'];
   }

   if ($a['one_way_fare'] != $b['one_way_fare']) {
      return $a['one_way_fare'] <=> $b['one_way_fare'];
   }

   return strnatcasecmp($a['destination'], $b['destination']);
});

return_fareのみでソート:

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

注:usort!で匿名関数を使用する必要はありません!

function cmp($a, $b) {
   return $a['return_fare'] <=> $b['return_fare'];
}

usort($data, 'cmp');


// Use a function inside a class:
class MyClass {
   public static function compare($a, $b) {
      return $a['return_fare'] <=> $b['return_fare'];
   }
}

usort($data, ['MyClass', 'compare']);

エルビス演算子(?:)を使用してこれらをチェーン化することもできます。

usort($data, function($a, $b) {
      return $a['return_fare'] <=> $b['return_fare'] ?:
             $a['one_way_fare'] <=> $b['one_way_fare'] ?:
             strnatcasecmp($a['destination'], $b['destination']);
});

この最後の例では、宇宙船オペレーター(<=>)とエルビスオペレーター(?:)を使用しました。プログラミングはうまくありませんか?

4
Westy92

ああ、私は再び自分の質問を解決することができた....

_function array_multi_sort($array, $on1,$on2, $order=SORT_ASC) 
{

    foreach($array as $key=>$value){
        $one_way_fares[$key] = $value[$on2];
        $return_fares[$key] = $value[$on1];
    }

    array_multisort($return_fares,$order,$one_way_fares,$order,$array);
}
_

問題は、array_multisort($return_fares,$order,$one_way_fares,$order,$array);の最後のパラメーター$ arrayを見逃したことです。

早く!

0
flyclassic