2012-06-19 18 views
7

Mam następującą tablicę:Sortuj tablicę używając array_multisort

$array = array(
     'note' => array(), 
     'year' => array('2011','2010', '2012'), 
     'type' => array('conference', 'journal', 'conference'), 
    ); 

i używam następującą funkcję aby posortować tablicę używając typ pola i kolejną tablicę:

function array_multisort_by_order(array $array, $by, array $order) 
{ 
    $order = array_flip($order); 
    $params[] = $array[$by]; 
    foreach($params[0] as &$v) $v = $order[$v]; 
    foreach($array as &$v) $params[] = &$v; unset($v); 
    call_user_func_array('array_multisort', $params); 
    return $array; 
} 

Kiedy zadzwonić pod następujące funkcja pojawia się następujący błąd:

$array = array_multisort_by_order($array, 'type', array('conference', 'journal')); 

print_r($array['type']); 

błąd:

Warning: array_multisort(): Array sizes are inconsistent. 

Wiem, że tablice są niespójne. Czy istnieje lepsza funkcja do użycia?

proszę sprawdzić: codepad

pożądany wynik:

Array 
(
[note] => Array 
    (
     [0] => 
     [1] => 
     [2] => 
    ) 

[year] => Array 
    (
     [0] => 2011 
     [1] => 2012 
     [2] => 2010 
    ) 

[type] => Array 
    (
     [0] => conference 
     [1] => conference 
     [2] => journal 
    ) 

) 

Przykład 2:

Array

$array = array(
     'note' => array([0]=>'test1', [1]=>'test2'), 
     'year' => array([0]=>'2011', [2]=>'2012'), 
     'type' => array([0]=>'conference',[1]=>'journal', [2]=>'conference'), 
    ); 

pożądany rezultat 2

Array 
(
[note] => Array 
    (
     [0] => test1 
     [1] => 
     [2] => tes2 
    ) 

[year] => Array 
    (
     [0] => 2011 
     [1] => 2012 
     [2] => 
    ) 

[type] => Array 
    (
     [0] => conference 
     [1] => conference 
     [2] => journal 
    ) 

) 

Dzięki

+3

Hm, sygnał wyjściowy będzie dokonać rzeczy bardziej przejrzyste niż wysyłanie go z kodu. Czy możesz to zapewnić? – Wrikken

+0

dodane pożądane wyjście! – glarkou

+0

OK, pozostało 1 pytanie: czy podmarki są zawsze puste lub o stałej długości, czy możemy oczekiwać podtablicy zawierającej 2 elementy, a jeśli tak, jak sobie z tym poradzić? Załóżmy, że nadal pasują one do pierwszych 2 elementów z innych subarrayów? – Wrikken

Odpowiedz

3

OK, więc jednym z pierwszych rozwiązań, które przychodzi do głowy jest dodanie w pustych wartości, aby były zgodne:

function array_multisort_by_order(array $array, $by, array $order) 
{ 
    $max = max(array_map('count',$array)); 
    //or, alternatively, depending on input (if there are no 'complete' subarrays): 
    //$max = max(array_map(function($arr){return max(array_keys($arr));},$array))+1; 

    //ADDITION: negative numeric keys: 
    $min = min(array_map(function($arr){return min(array_keys($arr));},$array)); 
    $width = $max - min(0,$min); 

    foreach($array as &$sub){ 
     // $addin = array_diff_key(array_fill(0,$max,null),$sub); 
     // $addin changed for negative keys: 
     $addin = array_diff_key(array_combine(range($min,$max),array_fill(0,$width,null)),$sub); 
     $sub = $addin + $sub; 
     ksort($sub); 
    } 
    $order = array_flip($order); 
    $params[] = $array[$by]; 
    foreach($params[0] as &$v) $v = $order[$v]; 
    foreach($array as &$v) $params[] = &$v; unset($v); 
    call_user_func_array('array_multisort', $params); 
    //no closeures here: 
    //foreach($array as &$sub) $sub = array_filter(function($a){return !is_null($a);},$sub); 
    $filter = create_function('$a','return !is_null($a);'); 
    foreach($array as &$sub) $sub = array_filter($sub,$filter); 
    return $array; 
} 
+0

Tak, stary, pomyślałem o tym. I zrobiłem http://codepad.org/4QcAoemv, ale tak naprawdę nie jest to dodawanie kluczy do oryginalnej tablicy. Dodatkowo istnieje sposób na pozbycie się tych "pustych" wartości po wykonaniu multisorta? – glarkou

+0

Ugh, kodek działa <5.3, nie może używać zamknięć ... Coś takiego masz na myśli: http://codepad.org/5kMHlRc6 – Wrikken

+0

Tak, to się uda. Jeśli możesz sprawdzić kod, spróbowałem użyć prostszej metody wypełniania brakujących kluczy 'foreach ($ tablica ['typ'] jako $ k => $ v) foreach ($ tablica jako $ element => $ a) { $ iterator = $ array [$ element]; if (! Isset ($ iterator [$ k])) { $ iterator [$ key] = ''; } } } '. Było to tymczasowe wypełnienie tablicy '$ iterator', a nie oryginalnej tablicy. Jeszcze raz dziękuję za pomoc. Wykorzystam Twoje rozwiązanie. Oznaczyłem to jako poprawne. Czy możesz edytować odpowiedź? – glarkou

Powiązane problemy