Как в PHP изменить ключ элемента массива?

У меня есть ассоциативный массив в форме key => value, где ключ - это числовое значение, однако это не последовательное числовое значение. Ключ на самом деле является идентификационным номером, а значение - числом. Это нормально для большинства случаев, однако мне нужна функция, которая получает удобочитаемое имя массива и использует его для ключа без изменения значения.

Я не видел функции, которая бы это делала, но предполагаю, что мне нужно предоставить старый ключ и новый ключ (оба из которых у меня есть) и преобразовать массив. Есть ли эффективный способ сделать это?


person Thomas Owens    schedule 27.10.2008    source источник
comment
См. Аналогичный stackoverflow.com/q/308703   -  person Peter Krauss    schedule 29.07.2013


Ответы (23)


Чтобы сделать это и сохранить порядок в массиве, нужно поместить ключи массива в отдельный массив, найти и заменить ключ в этом массиве, а затем снова объединить его со значениями.

Вот функция, которая делает именно это:

function change_key( $array, $old_key, $new_key ) {

    if( ! array_key_exists( $old_key, $array ) )
        return $array;

    $keys = array_keys( $array );
    $keys[ array_search( $old_key, $keys ) ] = $new_key;

    return array_combine( $keys, $array );
}
person DiverseAndRemote.com    schedule 23.01.2014
comment
Спасибо, это было действительно полезно, так как мне нужно было сохранить порядок массива. Я уже пробовал принятый ответ, прежде чем нашел эту страницу. - person gillytech; 02.02.2016
comment
Да уж, предпочитаю сохранять порядок в массиве, аккуратнее смотрится. - person Phil Cook; 24.03.2016
comment
Пришлось сохранить ключевой порядок, хороший, работал как шарм! - person Learner; 18.12.2018
comment
Не забывайте, если вам нужны выступления или сохранение порядка: stackoverflow.com/a/58619985/1617857 - person Léo Benoist; 30.10.2019

если ваш array создан на основе запроса к базе данных, вы можете изменить ключ прямо из оператора mysql:

вместо того

"select ´id´ from ´tablename´..."

используйте что-то вроде:

"select ´id´ **as NEWNAME** from ´tablename´..."
person Simon Franco    schedule 16.11.2010
comment
отличный ответ, очень ценно! - person DevMoutarde; 21.05.2019

Ответ KernelM хорош, но во избежание проблемы, поднятой Грегом в комментарии (конфликтующие ключи), безопаснее использовать новый массив.

$newarr[$newkey] = $oldarr[$oldkey];
$oldarr=$newarr;
unset($newarr);
person kjg    schedule 26.12.2010
comment
Это хорошее решение, если ваш массив имеет разумный размер. Если ваш массив потребляет более половины доступной памяти PHP, это не сработает. - person kingjeffrey; 08.03.2011
comment
@kingjeffrey, не совсем. Значения массива не будут дублироваться, пока они просто копируются без изменения. Например, если есть один массив, который содержит 10'000 элементов и потребляет 40 МБ памяти, его копирование потребует памяти, необходимой для хранения 10'000 только ссылок на уже существующие значения, а не копий значений. , поэтому, если 1 массив занимает 40 МБ, его копия может занимать 0,5 МБ (проверено). - person binaryLV; 19.07.2011

Вы можете использовать второй ассоциативный массив, который сопоставляет удобочитаемые имена с идентификаторами. Это также обеспечило бы отношения «Многие к 1». Затем сделайте что-нибудь вроде этого:

echo 'Widgets: ' . $data[$humanreadbleMapping['Widgets']];
person Tom Ritter    schedule 27.10.2008

Если вы хотите, чтобы позиция нового ключа массива была такой же, как и у старого, вы можете сделать это:

function change_array_key( $array, $old_key, $new_key) {
    if(!is_array($array)){ print 'You must enter a array as a haystack!'; exit; }
    if(!array_key_exists($old_key, $array)){
        return $array;
    }

    $key_pos = array_search($old_key, array_keys($array));
    $arr_before = array_slice($array, 0, $key_pos);
    $arr_after = array_slice($array, $key_pos + 1);
    $arr_renamed = array($new_key => $array[$old_key]);

    return $arr_before + $arr_renamed + $arr_after;
}
person spreadzz    schedule 27.01.2015

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

    $datos = array
    (
        '0' => array
            (
                'no' => 1,
                'id_maquina' => 1,
                'id_transaccion' => 1276316093,
                'ultimo_cambio' => 'asdfsaf',
                'fecha_ultimo_mantenimiento' => 1275804000,
                'mecanico_ultimo_mantenimiento' =>'asdfas',
                'fecha_ultima_reparacion' => 1275804000,
                'mecanico_ultima_reparacion' => 'sadfasf',
                'fecha_siguiente_mantenimiento' => 1275804000,
                'fecha_ultima_falla' => 0,
                'total_fallas' => 0,
            ),

        '1' => array
            (
                'no' => 2,
                'id_maquina' => 2,
                'id_transaccion' => 1276494575,
                'ultimo_cambio' => 'xx',
                'fecha_ultimo_mantenimiento' => 1275372000,
                'mecanico_ultimo_mantenimiento' => 'xx',
                'fecha_ultima_reparacion' => 1275458400,
                'mecanico_ultima_reparacion' => 'xx',
                'fecha_siguiente_mantenimiento' => 1275372000,
                'fecha_ultima_falla' => 0,
                'total_fallas' => 0,
            )
    );

вот функция:

function changekeyname($array, $newkey, $oldkey)
{
   foreach ($array as $key => $value) 
   {
      if (is_array($value))
         $array[$key] = changekeyname($value,$newkey,$oldkey);
      else
        {
             $array[$newkey] =  $array[$oldkey];    
        }

   }
   unset($array[$oldkey]);          
   return $array;   
}
person pajafumo    schedule 25.10.2012

Простое эталонное сравнение обоих решений.

Решение 1. Скопируйте и удалите (заказ потерян, но намного быстрее) https://stackoverflow.com/a/240676/1617857

<?php
$array = ['test' => 'value', ['etc...']];

$array['test2'] = $array['test'];
unset($array['test']);

Решение 2 Переименуйте ключ в https://stackoverflow.com/a/21299719/1617857

<?php
$array = ['test' => 'value', ['etc...']];

$keys = array_keys( $array );
$keys[array_search('test', $keys, true)] = 'test2';
array_combine( $keys, $array );

Контрольный показатель:

<?php
$array = ['test' => 'value', ['etc...']];


for ($i =0; $i < 100000000; $i++){
    // Solution 1
}


for ($i =0; $i < 100000000; $i++){
    // Solution 2
}

Полученные результаты:

php solution1.php  6.33s  user 0.02s system 99% cpu 6.356  total
php solution1.php  6.37s  user 0.01s system 99% cpu 6.390  total
php solution2.php  12.14s user 0.01s system 99% cpu 12.164 total
php solution2.php  12.57s user 0.03s system 99% cpu 12.612 total
person Léo Benoist    schedule 30.10.2019

Мне нравится решение KernelM, но мне нужно было что-то, что могло бы обрабатывать потенциальные конфликты ключей (где новый ключ может совпадать с существующим). Вот что я придумал:

function swapKeys( &$arr, $origKey, $newKey, &$pendingKeys ) {
    if( !isset( $arr[$newKey] ) ) {
        $arr[$newKey] = $arr[$origKey];
        unset( $arr[$origKey] );
        if( isset( $pendingKeys[$origKey] ) ) {
            // recursion to handle conflicting keys with conflicting keys
            swapKeys( $arr, $pendingKeys[$origKey], $origKey, $pendingKeys );
            unset( $pendingKeys[$origKey] );
        }
    } elseif( $newKey != $origKey ) {
        $pendingKeys[$newKey] = $origKey;
    }
}

Затем вы можете циклически перемещаться по массиву следующим образом:

$myArray = array( '1970-01-01 00:00:01', '1970-01-01 00:01:00' );
$pendingKeys = array();
foreach( $myArray as $key => $myArrayValue ) {
    // NOTE: strtotime( '1970-01-01 00:00:01' ) = 1 (a conflicting key)
    $timestamp = strtotime( $myArrayValue );
    swapKeys( $myArray, $key, $timestamp, $pendingKeys );
}
// RESULT: $myArray == array( 1=>'1970-01-01 00:00:01', 60=>'1970-01-01 00:01:00' )
person kingjeffrey    schedule 08.03.2011

Вот вспомогательная функция для этого:

/**
 * Helper function to rename array keys.
 */
function _rename_arr_key($oldkey, $newkey, array &$arr) {
    if (array_key_exists($oldkey, $arr)) {
        $arr[$newkey] = $arr[$oldkey];
        unset($arr[$oldkey]);
        return TRUE;
    } else {
        return FALSE;
    }
}

довольно на основе ответа @KernelM.

Использование:

_rename_arr_key('oldkey', 'newkey', $my_array);

Он вернет true при успешном переименовании, в противном случае - false.

person kenorb    schedule 30.11.2015
comment
Имейте в виду, что это изменяет порядок массива (переименованный элемент ключа будет в конце массива, а не в той же позиции в массиве, как это было изначально). Также я обычно не начинал имя функции с подчеркивания (которое традиционно используется для обозначения специальных внутренних функций). - person orrd; 18.06.2016

Легкие вещи:

эта функция примет целевой $ hash, а $ replacements также является хешем, содержащим newkey => oldkey ассоциации.

Эта функция сохранит исходный порядок, но может быть проблематичной для очень больших (например, более 10 тыс. Записей) массивов, касающихся производительности и памяти.

function keyRename(array $hash, array $replacements) {
    $new=array();
    foreach($hash as $k=>$v)
    {
        if($ok=array_search($k,$replacements))
            $k=$ok;
        $new[$k]=$v;
    }
    return $new;    
}

эта альтернативная функция будет делать то же самое, с гораздо большей производительностью и использованием памяти, за счет потери исходного порядка (что не должно быть проблемой, поскольку это хеш-таблица!)

function keyRename(array $hash, array $replacements) {

    foreach($hash as $k=>$v)
        if($ok=array_search($k,$replacements))
        {
          $hash[$ok]=$v;
          unset($hash[$k]);
        }

    return $hash;       
}
person Nadir    schedule 20.07.2015

этот код поможет поменять старый ключ на новый

$i = 0;
$keys_array=array("0"=>"one","1"=>"two");

$keys = array_keys($keys_array);

for($i=0;$i<count($keys);$i++) {
    $keys_array[$keys_array[$i]]=$keys_array[$i];
    unset($keys_array[$i]);
}
print_r($keys_array);

отображать как

$keys_array=array("one"=>"one","two"=>"two");
person karthikeyan ganesan    schedule 22.06.2015

Вы можете использовать эту функцию на основе array_walk:

function mapToIDs($array, $id_field_name = 'id')
{
    $result = [];
    array_walk($array, 
        function(&$value, $key) use (&$result, $id_field_name)
        {
            $result[$value[$id_field_name]] = $value;
        }
    );
    return $result;
}

$arr = [0 => ['id' => 'one', 'fruit' => 'apple'], 1 => ['id' => 'two', 'fruit' => 'banana']];
print_r($arr);
print_r(mapToIDs($arr));

Это дает:

Array(
    [0] => Array(
        [id] => one
        [fruit] => apple
    )
    [1] => Array(
        [id] => two
        [fruit] => banana
    )
)

Array(
    [one] => Array(
        [id] => one
        [fruit] => apple
    )
    [two] => Array(
        [id] => two
        [fruit] => banana
    )
)
person Alekzander    schedule 09.09.2018

это работает для переименования первого ключа:

$a = ['catine' => 'cat', 'canine'  => 'dog'];
$tmpa['feline'] = $a['catine'];
unset($a['catine']);
$a = $tmpa + $a;

затем print_r ($ a) визуализирует восстановленный массив по порядку:

Array
(
    [feline] => cat
    [canine] => dog
)

это работает для переименования произвольного ключа:

$a = ['canine'  => 'dog', 'catine' => 'cat', 'porcine' => 'pig']
$af = array_flip($a)
$af['cat'] = 'feline';
$a = array_flip($af)

print_r ($ а)

Array
(
    [canine] => dog
    [feline] => cat
    [porcine] => pig
)

обобщенная функция:

function renameKey($oldkey, $newkey, $array) {
    $val = $array[$oldkey];
    $tmp_A = array_flip($array);
    $tmp_A[$val] = $newkey;

    return array_flip($tmp_A);
}
person wmmso    schedule 15.05.2016

Если вы хотите заменить сразу несколько ключей (с сохранением порядка):

/**
 * Rename keys of an array
 * @param array $array (asoc)
 * @param array $replacement_keys (indexed)
 * @return array
 */
function rename_keys($array, $replacement_keys)  {
      return array_combine($replacement_keys, array_values($array));
}

Использование:

$myarr = array("a" => 22, "b" => 144, "c" => 43);
$newkeys = array("x","y","z");
print_r(rename_keys($myarr, $newkeys));
//must return: array("x" => 22, "y" => 144, "z" => 43);
person lepe    schedule 13.07.2016

Есть альтернативный способ изменить ключ элемента массива при работе с полным массивом - без изменения порядка массива. Просто скопируйте массив в новый массив.

Например, я работал со смешанным многомерным массивом, который содержал индексированные и ассоциативные ключи, и я хотел заменить целочисленные ключи их значениями, не нарушая порядок.

Я сделал это, переключив ключ / значение для всех записей числового массива - здесь: ['0' => 'foo']. Обратите внимание, что порядок не поврежден.

<?php
$arr = [
    'foo',
    'bar'=>'alfa',
    'baz'=>['a'=>'hello', 'b'=>'world'],
];

foreach($arr as $k=>$v) {
    $kk = is_numeric($k) ? $v : $k;
    $vv = is_numeric($k) ? null : $v;
    $arr2[$kk] = $vv;
}

print_r($arr2);

Выход:

Array (
    [foo] => 
    [bar] => alfa
    [baz] => Array (
            [a] => hello
            [b] => world
        )
)
person Kristoffer Bohmann    schedule 28.03.2017

лучший способ - использовать ссылку, а не использовать unset (что делает еще один шаг для очистки памяти)

$tab = ['two' => [] ];

решение:

$tab['newname'] = & $tab['two'];

у вас есть один оригинал и одна ссылка с новым именем.

или если вы не хотите иметь два имени в одном значении, хорошо сделать другую вкладку и foreach по ссылке

foreach($tab as $key=> & $value) {
    if($key=='two') { 
        $newtab["newname"] = & $tab[$key];
     } else {
        $newtab[$key] = & $tab[$key];
     }
}

Итерация лучше для ключей, чем клонирование всего массива, и очистка старого массива, если у вас длинные данные, такие как 100 строк +++ и т. Д.

person Kamil Dąbrowski    schedule 31.05.2019

Эта функция будет переименовывать ключ массива, сохраняя его позицию, путем комбинирования с поиском по индексу.

function renameArrKey($arr, $oldKey, $newKey){
    if(!isset($arr[$oldKey])) return $arr; // Failsafe
    $keys = array_keys($arr);
    $keys[array_search($oldKey, $keys)] = $newKey;
    $newArr = array_combine($keys, $arr);
    return $newArr;
}

Использование:

$arr = renameArrKey($arr, 'old_key', 'new_key');
person Grant    schedule 19.03.2020

Хм, я раньше не тестировал, но думаю, что этот код работает

function replace_array_key($data) {
    $mapping = [
        'old_key_1' => 'new_key_1',
        'old_key_2' => 'new_key_2',
    ];

    $data = json_encode($data);
    foreach ($mapping as $needed => $replace) {
        $data = str_replace('"'.$needed.'":', '"'.$replace.'":', $data);
    }

    return json_decode($data, true);
}
person Frank Vu    schedule 16.01.2017
comment
Json кодировать и декодировать? Это действительно плохой ответ. - person adamkonrad; 05.08.2017

Тот, который упорядочивает консервы, это просто понять:

function rename_array_key(array $array, $old_key, $new_key) {
  if (!array_key_exists($old_key, $array)) {
      return $array;
  }
  $new_array = [];
  foreach ($array as $key => $value) {
    $new_key = $old_key === $key
      ? $new_key
      : $key;
    $new_array[$new_key] = $value;
  }
  return $new_array;
}
person Andrew    schedule 15.08.2018

Вы можете написать простую функцию, которая применяет обратный вызов к ключам данного массива. Подобно array_map

<?php
function array_map_keys(callable $callback, array $array) {
    return array_merge([], ...array_map(
        function ($key, $value) use ($callback) { return [$callback($key) => $value]; },
        array_keys($array),
        $array
    ));
}

$array = ['a' => 1, 'b' => 'test', 'c' => ['x' => 1, 'y' => 2]];
$newArray = array_map_keys(function($key) { return 'new' . ucfirst($key); }, $array);

echo json_encode($array); // {"a":1,"b":"test","c":{"x":1,"y":2}}
echo json_encode($newArray); // {"newA":1,"newB":"test","newC":{"x":1,"y":2}}

Вот суть https://gist.github.com/vardius/650367e15abfb58bcd72ca47eff096ca#file-array_map_keys-php.

person vardius    schedule 28.06.2019

Эта базовая функция обрабатывает замену ключей массива и сохраняет массив в исходном порядке ...

public function keySwap(array $resource, array $keys)
{
    $newResource = [];

    foreach($resource as $k => $r){
        if(array_key_exists($k,$keys)){
            $newResource[$keys[$k]] = $r;
        }else{
            $newResource[$k] = $r;
        }
    }

    return $newResource;
}

Затем вы могли бы перебрать и поменять, например, все клавиши 'a' на 'z' ...

$inputs = [
  0 => ['a'=>'1','b'=>'2'],
  1 => ['a'=>'3','b'=>'4']
]

$keySwap = ['a'=>'z'];

foreach($inputs as $k=>$i){
    $inputs[$k] = $this->keySwap($i,$keySwap);
}
person MikeyJ    schedule 30.10.2019
comment
Спасибо, это именно то, что я искал. - person Phani Shashank; 20.07.2021

person    schedule
comment
Просто будьте осторожны, чтобы 1) у двух ключей не было одинаковой удобочитаемой версии 2) Ни одна из удобочитаемых версий не была числами. - person Greg; 27.10.2008
comment
Также это предположительно изменит порядок массива, о чем вам, возможно, придется быть осторожным. Даже ассоциативные массивы в PHP упорядочены, и иногда этот порядок используется. - person Robin Winslow; 09.02.2012
comment
Да, отличный момент, Робин. Есть ли способ сохранить такой же порядок? Или для этого вам нужно создать новый массив? - person Simon East; 28.06.2012
comment
Бонусный вопрос: как изменить идентификатор, но сохранить порядок в массиве? - person Petr Peller; 12.12.2012
comment
если значение ключа не меняется, вы удалите элемент массива. Вы можете проверить это. - person Peeech; 10.03.2013
comment
Ответ Тома Риттера сохраняет порядок. - person Jorge Orpinel Pérez; 10.04.2014
comment
Я не могу ответить на комментарий Петра Пеллера (который хочет изменить ключ, но сохранить порядок в массиве), потому что моей репутации недостаточно. Я нашел его в ... stackoverflow.com/questions/10182684/ может быть полезно :) - person Mei; 06.06.2014
comment
Также существует проблема, заключающаяся в том, что новый ключ может перезаписать существующий старый ключ и его значения, что приводит к удалению старого ключевого элемента из массива. В этом случае вам нужно использовать два массива. - person TeeJay; 29.07.2015
comment
это также сбросит значение, поэтому не лучший вариант для ассоциативного артефакта. - person Giedrius; 22.05.2017
comment
сэкономил час своего времени и подарил мне красивый код! Спасибо:) - person Sergej Fomin; 20.07.2017
comment
Это так очевидно / спасибо! - person Jonas de Herdt; 27.10.2017
comment
Спасибо, это мне немного помогло - person icecub; 19.10.2018
comment
Назначьте по ссылке, чтобы избежать копирования значения: $ arr [$ newkey] = & $ arr [$ oldkey]; - person andres101; 17.01.2019

person    schedule
comment
Я люблю свои функции массива. Я собирался предложить это как хороший однострочный вариант для переименования всех ключей и поддержания порядка массива, но вместо этого я рекомендую ваш. - person Autumn Leonard; 17.04.2018
comment
Красиво и плотно. Если вы работаете с большим массивом и не хотите менять все ключи, строка в функции карты становится return isset($renameMap[$el]) ? $renameMap[$el] : $el; - person Jerry; 02.04.2021