В PHP, как вы меняете ключ элемента массива?
У меня есть ассоциативный массив в виде key => value
где ключ-это числовое значение, однако это не последовательное числовое значение. Ключ идентификатор, а значение-это количество. Это нормально для большинства экземпляров, однако мне нужна функция, которая получает удобочитаемое имя массива и использует его для ключа, не изменяя значение.
Я не видел функции, которая делает это, но я предполагаю, что мне нужно предоставить старый ключ и новый ключ (оба из которых я есть) и преобразовать массив. Есть ли эффективный способ сделать это?
18 ответов:
способ, которым вы бы сделали это и сохранили порядок массива, заключается в том, чтобы поместить ключи массива в отдельный массив, найти и заменить ключ в этом массиве, а затем объединить его со значениями.
вот функция, которая делает именно это:
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 ); }
если
array
строится из запроса базы данных, вы можете изменить ключ непосредственно изmysql
о себе:вместо
"select ´id´ from ´tablename´..."
использовать что-то вроде:
"select ´id´ **as NEWNAME** from ´tablename´..."
ответ от KernelM хорош, но чтобы избежать проблемы, поднятой Грегом в комментарии (конфликтующие ключи), использование нового массива было бы безопаснее
$newarr[$newkey] = $oldarr[$oldkey]; $oldarr=$newarr; unset($newarr);
вы могли бы использовать второй ассоциативный массив, который сопоставляет понятные имена для идентификаторов. Это также будет служить многие до 1 отношения. Затем сделайте что-то вроде этого:
echo 'Widgets: ' . $data[$humanreadbleMapping['Widgets']];
Если вы также хотите, чтобы позиция нового ключа массива была такой же, как и старая, вы можете сделать это:
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; }
Если Ваш массив рекурсивный, вы можете использовать эту функцию: проверьте эти данные:
$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; }
мне нравится решение 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' )
вот вспомогательная функция для достижения этого:
/** * 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);
вернет правда при успешном переименовании, в противном случае ложные.
легкий материал:
эта функция будет принимать целевой $ hash и $replacements также хэш, содержащий newkey=>oldkey объединений.
эта функция будет сохранить исходный порядок, но может быть проблематичным для очень больших (например, выше 10k записей) массивов относительно производительность и память.
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; }
$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 ]; */
этот код поможет изменить старый ключ на новый
$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");
это работает для переименования первого ключа:
$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 ($a)
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); }
Если вы хотите заменить сразу несколько ключей (сохраняя порядок):
/** * 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);
существует альтернативный способ изменения ключа элемента массива при работе с полным массивом-без изменения порядка массива. Это просто скопировать массив в новый массив.
Я сделал это путем переключения ключа / значения для всех записей числового массива - здесь: ['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 ) )
Хм, я не тестирую раньше, но я думаю, что этот код работает
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); }
тот, который сохраняет порядок, который просто понять:
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; }
вы можете использовать эту функцию на основе 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 ) )