Як «вирівняти» багатовимірний масив до простого в PHP?


83

Це, мабуть, питання для початківців, але я вже довше переглядаю документацію і не можу знайти жодного рішення. Я думав, що можу використовувати implode для кожного виміру, а потім повернути ці рядки назад разом, str_splitщоб створити новий простий масив. Однак я ніколи не знаю, чи шаблон об'єднання також не є значеннями, і тому після виконання str_splitмоїх початкових значень може зламатися.

Чи є щось на зразок combine($array1, $array2)масивів усередині багатовимірного масиву?


Будь ласка , перевірте це посилання для вирішення : stackoverflow.com/questions/14951811 / ...
Prasanth Bendra

1
Ще одне гарне довідкове запитання з, можливо, кращими відповідями: Як зрівняти багатовимірний масив?
hakre

Відповіді:


42

Використовуйте array_walk_recursive

<?php

$aNonFlat = array(
    1,
    2,
    array(
        3,
        4,
        5,
        array(
            6,
            7
        ),
        8,
        9,
    ),
    10,
    11
);

$objTmp = (object) array('aFlat' => array());

array_walk_recursive($aNonFlat, create_function('&$v, $k, &$t', '$t->aFlat[] = $v;'), $objTmp);

var_dump($objTmp->aFlat);

/*
array(11) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
  [3]=>
  int(4)
  [4]=>
  int(5)
  [5]=>
  int(6)
  [6]=>
  int(7)
  [7]=>
  int(8)
  [8]=>
  int(9)
  [9]=>
  int(10)
  [10]=>
  int(11)
}
*/

?>

Протестовано з PHP 5.5.9-1ubuntu4.24 (cli) (побудовано: 16 березня 2018 12:32:06)


Хто-небудь знає, чому це не працює, якщо я не використовую (амортизований) пропускний час за посиланням. тобто array_walk_recursive ($ array, create_function ('& $ v, $ k, & $ t', '$ t [] = $ v;'), & $ сплощений); Визначення функції правильно визначено як передачу за посиланням. але не працює, якщо я не проходжу посилання під час дзвінка.
jskulski

2
Об'єкти @jskilski ( $objTmpу цьому прикладі) передаються за посиланням автоматично; масиви не є. Спробуйте використати анонімну функцію ( php.net/manual/en/functions.anonymous.php ) замість create_function.
dave1010

1
це не працює в php 5.3.3 через помилку в array_walk_recursive - bugs.php.net/bug.php?id=52719
crazyphoton

1
@crazyphoton The kink також говоритьThis bug has been fixed in SVN.
Luc M

10
Чому в цій відповіді згадується використання array_values()? Я взагалі не бачу використання цієї функції, яка бере участь у відповіді.
thomasrutter

132
$array  = your array

$result = call_user_func_array('array_merge', $array);

echo "<pre>";
print_r($result);

REF: http://php.net/manual/en/function.call-user-func-array.php

Ось ще одне рішення (працює з багатовимірним масивом):

function array_flatten($array) {

   $return = array();
   foreach ($array as $key => $value) {
       if (is_array($value)){ $return = array_merge($return, array_flatten($value));}
       else {$return[$key] = $value;}
   }
   return $return;

}

$array  = Your array

$result = array_flatten($array);

echo "<pre>";
print_r($result);

Ця відповідь набагато швидша за прийняту відповідь.
Roham Rafii

8
Оскільки php5.3 тепер ви можете використовувати оператор splat: $result = array_merge(...$array); php.net/manual/en/…
Redzarf

1
Ваша перша відповідь не працює з багатовимірним масивом. 3v4l.org/tY8vD
dearsina

54

Це один рядок, SUPER простий у використанні:

$result = array();
array_walk_recursive($original_array,function($v) use (&$result){ $result[] = $v; });

Це дуже легко зрозуміти всередині анонімної функції / закриття. $v- цінність вашого $original_array.


4
Це єдиний, який працював у мене в дворівневому масиві.
Ciprian Tepes,

18
// $array = your multidimensional array

$flat_array = array();

foreach(new RecursiveIteratorIterator(new RecursiveArrayIterator($array)) as $k=>$v){

$flat_array[$k] = $v;

}

Також задокументовано: http://www.phpro.org/examples/Flatten-Array.html


2
Примітка: Використовуйте лише для масивів примітивів. "RecursiveArrayIterator розглядає всі об'єкти як такі, що мають дітей, і намагається повернутися до них." php.net/manual/en/class.recursivearrayiterator.php#106519
ReactiveRaven

@hakre: +1 погодився: додавання iterator_to_array()до цієї відповіді заперечуватиме потребу в foreachциклі. Це може бути проста однофункціональна функція. (хоча і дещо довгий однорядковий)
SDC

2
Я знаю, що це старе, але все ще корисне, однак $ k потрібно замінити чимось унікальним, наприклад, лічильником. Просто використання $ k призводить до видалення елементів, якщо імена у внутрішніх масивах збігаються з основним.
Austin Best

12

Якщо у вас спеціально є масив масивів, який не перевищує глибину одного рівня (варіант використання, який я вважаю загальним), ви можете піти з array_mergeоператором splat.

<?php

$notFlat = [[1,2],[3,4]];
$flat = array_merge(...$notFlat);
var_dump($flat);

Вихід:

array(4) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
  [3]=>
  int(4)
}

Оператор splat ефективно змінює масив масивів на список масивів як аргументи для array_merge.


Це виглядає як найкраща відповідь для мене. Він не буде працювати зі строковими клавішами, але його можна легко змінити, щоб це зробити:$flat = array_merge( array_keys( $notFlat ), ...array_values( $notFlat ) );
Ян Данн,

6

За допомогою PHP 7 ви можете використовувати генератори та делегування генератора ( yield from) для згладжування масиву:

function array_flatten_iterator (array $array) {
    foreach ($array as $value) {
        if (is_array($value)) {
            yield from array_flatten_iterator($value);
        } else {
            yield $value;
        }
    }
}

function array_flatten (array $array) {
    return iterator_to_array(array_flatten_iterator($array), false);
}

Приклад:

$array = [
    1,
    2,
    [
        3,
        4,
        5,
        [
            6,
            7
        ],
        8,
        9,
    ],
    10,
    11,
];    

var_dump(array_flatten($array));

http://3v4l.org/RU30W


5
function flatten_array($array, $preserve_keys = 0, &$out = array()) {
    # Flatten a multidimensional array to one dimension, optionally preserving keys.
    #
    # $array - the array to flatten
    # $preserve_keys - 0 (default) to not preserve keys, 1 to preserve string keys only, 2 to preserve all keys
    # $out - internal use argument for recursion
    foreach($array as $key => $child)
        if(is_array($child))
            $out = flatten_array($child, $preserve_keys, $out);
        elseif($preserve_keys + is_string($key) > 1)
            $out[$key] = $child;
        else
            $out[] = $child;
    return $out;
}

Вибачте, але, схоже, він не обробляє багатовимірні масиви належним чином - Демонстрація
Альваро Гонсалес

5

Інший метод із коментарів користувачів PHP (спрощений) і тут :

function array_flatten_recursive($array) { 
   if (!$array) return false;
   $flat = array();
   $RII = new RecursiveIteratorIterator(new RecursiveArrayIterator($array));
   foreach ($RII as $value) $flat[] = $value;
   return $flat;
}

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

$array = array( 
    'A' => array('B' => array( 1, 2, 3)), 
    'C' => array(4, 5) 
); 
print_r(array_flatten_recursive($array)); 

#Returns: 
Array ( 
    [0] => 1 
    [1] => 2 
    [2] => 3 
    [3] => 4 
    [4] => 5 
)

3
Примітка: Використовуйте лише для масивів примітивів. "RecursiveArrayIterator розглядає всі об'єкти як такі, що мають дітей, і намагається повернутися до них." php.net/manual/en/class.recursivearrayiterator.php#106519
ReactiveRaven

4

Нерекурсивне рішення (але руйнує порядок):

function flatten($ar) {
    $toflat = array($ar);
    $res = array();

    while (($r = array_shift($toflat)) !== NULL) {
        foreach ($r as $v) {
            if (is_array($v)) {
                $toflat[] = $v;
            } else {
                $res[] = $v;
            }
        }
    }

    return $res;
}

4

У PHP> = 5.3 та на основі відповіді Luc M (першої) ви можете скористатися подібними закриттями

array_walk_recursive($aNonFlat, function(&$v, $k, &$t){$t->aFlat[] = $v;}, $objTmp);

Мені це подобається, тому що мені не потрібно оточувати код функції лапками, як при використанні create_function ()


1
якщо ви використовуєте анонімні функції, ви можете скористатися захопленою змінною закриття, а не цим objTempматеріалом
user102008,

1
У PHP5.3.3 є помилка, яка спричиняє збій - bugs.php.net/bug.php?id=52719
crazyphoton

2

Використання функцій вищого порядку (примітка: я використовую вбудовані анонімні функції , які з’явилися в PHP 5.3):

function array_flatten($array) {
    return array_reduce(
        $array,
        function($prev, $element) {
            if (!is_array($element))
                $prev[] = $element;
            else
                $prev = array_merge($prev, array_flatten($element));
            return $prev;
        },
        array()
    );
}

1

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

# Flatten a multidimensional array to one dimension, optionally preserving keys.
# $array - the array to flatten
# $preserve_keys - 0 (default) to not preserve keys, 1 to preserve string keys only, 2 to preserve all keys
# $out - internal use argument for recursion

function flatten_array($array, $preserve_keys = 2, &$out = array(), &$last_subarray_found) 
{
        foreach($array as $key => $child)
        {
            if(is_array($child))
            {
                $last_subarray_found = $key;
                $out = flatten_array($child, $preserve_keys, $out, $last_subarray_found);
            }
            elseif($preserve_keys + is_string($key) > 1)
            {
                if ($last_subarray_found)
                {
                    $sfinal_key_value = $last_subarray_found . "_" . $key;
                }
                else
                {
                    $sfinal_key_value = $key;
                }
                $out[$sfinal_key_value] = $child;
            }
            else
            {
                $out[] = $child;
            }
        }

        return $out;
}

Example:
$newarraytest = array();
$last_subarray_found = "";
$this->flatten_array($array, 2, $newarraytest, $last_subarray_found);

1
/*consider $mArray as multidimensional array and $sArray as single dimensional array
this code will ignore the parent array
*/

function flatten_array2($mArray) {
    $sArray = array();

    foreach ($mArray as $row) {
        if ( !(is_array($row)) ) {
            if($sArray[] = $row){
            }
        } else {
            $sArray = array_merge($sArray,flatten_array2($row));
        }
    }
    return $sArray;
}

1

Ви можете спробувати це:

function flat_an_array($a)
{
    foreach($a as $i)
    {
        if(is_array($i)) 
        {
            if($na) $na = array_merge($na,flat_an_array($i));
            else $na = flat_an_array($i);
        }
        else $na[] = $i;
    }
    return $na;
}

1

Якщо ви гаразд втрачаєте ключі масиву, ви можете згладити багатовимірний масив, використовуючи рекурсивне закриття як зворотний виклик, який використовує array_values ​​(), переконавшись, що цей зворотний виклик є параметром для array_walk (), як показано нижче.

<?php  

$array = [1,2,3,[5,6,7]];
$nu_array = null;
$callback = function ( $item ) use(&$callback, &$nu_array) {
    if (!is_array($item)) {
    $nu_array[] = $item;
    }
    else
    if ( is_array( $item ) ) {
     foreach( array_values($item) as $v) {
         if ( !(is_array($v))) {
             $nu_array[] = $v;
         }
         else
         { 
             $callback( $v );
         continue;
         }    
     }
    }
};

array_walk($array, $callback);
print_r($nu_array);

Єдиним недоліком попереднього прикладу є те, що він передбачає написання набагато більше коду, ніж наступне рішення, яке використовує array_walk_recursive () разом із спрощеним зворотним викликом:

<?php  

$array = [1,2,3,[5,6,7]];

$nu_array = [];
array_walk_recursive($array, function ( $item ) use(&$nu_array )
                     {
                         $nu_array[] = $item;
                     }
);
print_r($nu_array);

Дивіться код в реальному часі

Цей приклад видається кращим перед попереднім, приховуючи деталі про те, як значення витягуються з багатовимірного масиву. Звичайно, ітерація трапляється, але чи тягне вона за собою рекурсію або структуру (и) управління, ви дізнаєтесь лише з перегляду array.c . Оскільки функціональне програмування зосереджується на вхідних і вихідних даних, а не на дрібницях отримання результату, безумовно, можна не турбуватися про те, як відбувається закулісна ітерація, тобто до тих пір, поки перспективний роботодавець не поставить таке питання.


1

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

Ось зразок.

$converted = http_build_query($data);
$rows = explode('&', $converted);
$output = array();
foreach($rows AS $k => $v){
   list($kk, $vv) = explode('=', $v);
   $output[ urldecode($kk) ] =  urldecode($vv);
}
return $output;

1

Вибачте за некробупінг, але жодна з наданих відповідей не зробила того, що я інтуїтивно розумів як "згладжування багатовимірного масиву". А саме цей випадок:

[
  'a' => [
    'b' => 'value',
  ]
]

всі запропоновані рішення згладять це просто ['value'], але при цьому втрачається інформація про ключ і глибину, плюс якщо у вас є інший ключ "b" де-небудь ще, він перезапише їх.

Я хотів отримати такий результат:

[
  'a_b' => 'value',
]

array_walk_recursive не передає інформацію про ключ, який він наразі повторюється, тому я зробив це з простою рекурсією:

function flatten($array, $prefix = '') {
    $return = [];
    foreach ($array as $key => $value) {
        if (is_array($value)) {
            $return = array_merge($return, flatten($value, $prefix . $key . '_'));
        } else {
            $return[$prefix . $key] = $value;
        }
    }
    return $return;
}

Змініть префікс $ та роздільник '_' на свій смак.

Дитячий майданчик тут: https://3v4l.org/0B8hf



0

Простий підхід .. Перегляньте це через рекурсію ..

<?php

function flatten_array($simple){
static $outputs=array();
foreach ( $simple as $value)
{
if(is_array($value)){
    flatten_array($value);
}
else{
    $outputs[]=$value;
}

}
return $outputs;
}

$eg=['s'=>['p','n'=>['t']]];
$out=flatten_array($eg);
print_r($out);

?>

Чому staticдля цього завдання використовується потенційно погана ідея? Ненавмисне збереження даних. Це, безсумнівно, застане програмістів зненацька, якщо вони не знають / очікують такої поведінки. Подивіться на цю демонстрацію .
mickmackusa

Ви опублікували відповідь "лише код" - це низьке значення для StackOverflow, оскільки вони не можуть навчити ОП та майбутніх дослідників. Приділіть трохи часу, щоб покращити цю відповідь, включивши, як працює ваша відповідь, і чому ви вважаєте, що це краща ідея порівняно з попередніми відповідями.
mickmackusa

0

Хтось може виявити це корисним, у мене виникла проблема згладжування масиву в якомусь вимірі, я б назвав його останнім виміром, наприклад, якщо у мене є такий масив:

array (
  'germany' => 
  array (
    'cars' => 
    array (
      'bmw' => 
      array (
        0 => 'm4',
        1 => 'x3',
        2 => 'x8',
      ),
    ),
  ),
  'france' => 
  array (
    'cars' => 
    array (
      'peugeot' => 
      array (
        0 => '206',
        1 => '3008',
        2 => '5008',
      ),
    ),
  ),
)

Або:

array (
  'earth' => 
  array (
    'germany' => 
    array (
      'cars' => 
      array (
        'bmw' => 
        array (
          0 => 'm4',
          1 => 'x3',
          2 => 'x8',
        ),
      ),
    ),
  ),
  'mars' => 
  array (
    'france' => 
    array (
      'cars' => 
      array (
        'peugeot' => 
        array (
          0 => '206',
          1 => '3008',
          2 => '5008',
        ),
      ),
    ),
  ),
)

Для обох цих масивів, коли я викликаю метод нижче, я отримую результат:

array (
  0 => 
  array (
    0 => 'm4',
    1 => 'x3',
    2 => 'x8',
  ),
  1 => 
  array (
    0 => '206',
    1 => '3008',
    2 => '5008',
  ),
)

Отже, я вирівнююсь до останнього виміру масиву, який повинен залишатися незмінним, метод нижче може бути реконструйований, щоб фактично зупинитися на будь-якому рівні:

function flattenAggregatedArray($aggregatedArray) {
    $final = $lvls = [];
    $counter = 1;
    $lvls[$counter] = $aggregatedArray;


    $elem = current($aggregatedArray);

    while ($elem){
        while(is_array($elem)){
            $counter++;
            $lvls[$counter] = $elem;
            $elem =  current($elem);
        }

        $final[] = $lvls[$counter];
        $elem = next($lvls[--$counter]);
        while ( $elem  == null){
            if (isset($lvls[$counter-1])){
                $elem = next($lvls[--$counter]);
            }
            else{
                return $final;
            }
        }
    }
}

-1

Якщо вас цікавлять лише значення для одного конкретного ключа, вам може виявитися корисним такий підхід:

function valuelist($array, $array_column) {
    $return = array();
    foreach($array AS $row){
        $return[]=$row[$array_column];
    };
    return $return;
};

Приклад:

Дано $ get_role_action =

array(3) {
  [0]=>
  array(2) {
    ["ACTION_CD"]=>
    string(12) "ADD_DOCUMENT"
    ["ACTION_REASON"]=>
    NULL
  }
  [1]=>
  array(2) {
    ["ACTION_CD"]=>
    string(13) "LINK_DOCUMENT"
    ["ACTION_REASON"]=>
    NULL
  }
  [2]=>
  array(2) {
    ["ACTION_CD"]=>
    string(15) "UNLINK_DOCUMENT"
    ["ACTION_REASON"]=>
    NULL
  }
}

ніж $variables['role_action_list']=valuelist($get_role_action, 'ACTION_CD');це призведе до:

$variables["role_action_list"]=>
  array(3) {
    [0]=>
    string(12) "ADD_DOCUMENT"
    [1]=>
    string(13) "LINK_DOCUMENT"
    [2]=>
    string(15) "UNLINK_DOCUMENT"
  }

Звідти ви можете виконувати пошук цінності таким чином:

if( in_array('ADD_DOCUMENT', $variables['role_action_list']) ){
    //do something
};

Це PHP-нокаут функції CFML з однойменною назвою.
Jeromy French

Я проголосував за те, що це правильна відповідь на неправильне питання.
mickmackusa

-1

будь-що з цього не спрацювало для мене ... тому довелося запустити це самому. працює просто чудово:

function arrayFlat($arr){
$out = '';
    foreach($arr as $key => $value){

        if(!is_array($value)){
            $out .= $value.',';
        }else{
            $out .= $key.',';
            $out .= arrayFlat($value);
        }

    }
    return trim($out,',');
}


$result = explode(',',arrayFlat($yourArray));
echo '<pre>';
print_r($result);
echo '</pre>';

Ця відповідь лише для коду не працює за бажанням. 3v4l.org/U3bfp <- доказ Це причина мого голосування проти.
mickmackusa

-1

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

function _flatten_array($arr) {
  while ($arr) {
    list($key, $value) = each($arr); 
    is_array($value) ? $arr = $value : $out[$key] = $value;
    unset($arr[$key]);
  }
  return (array)$out;
}

Я підтримав цю відповідь, оскільки вона не працює в жодній версії. 3v4l.org/7cO9N (Доказ) Також each()застаріло від php7.2.
mickmackusa
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.