В PHP, как вы меняете ключ элемента массива? - PullRequest
318 голосов
/ 27 октября 2008

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

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

Ответы [ 19 ]

505 голосов
/ 27 октября 2008
$arr[$newkey] = $arr[$oldkey];
unset($arr[$oldkey]);
81 голосов
/ 23 января 2014

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

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

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 );
}
49 голосов
/ 16 ноября 2010

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

вместо

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

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

"select ´id´ **as NEWNAME** from ´tablename´..."
18 голосов
/ 26 декабря 2010

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

$newarr[$newkey] = $oldarr[$oldkey];
$oldarr=$newarr;
unset($newarr);
16 голосов
/ 27 октября 2008

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

echo 'Widgets: ' . $data[$humanreadbleMapping['Widgets']];
10 голосов
/ 27 января 2015

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

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;
}
6 голосов
/ 30 ноября 2015

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

/**
 * 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 .

6 голосов
/ 25 октября 2012

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

    $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;   
}
6 голосов
/ 08 марта 2011

Мне нравится решение 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' )
6 голосов
/ 05 апреля 2016
$array = [
    'old1' => 1
    'old2' => 2
];

$renameMap = [
    'old1' => 'new1',   
    'old2' => 'new2'
];

$array = array_combine(array_map(function($el) use ($renameMap) {
    return $renameMap[$el];
}, array_keys($array)), array_values($array));

/*
$array = [
    'new1' => 1
    'new2' => 2
];
*/
...