ChatGPT解决这个技术问题 Extra ChatGPT

根据另一个查找/映射数组替换数组中的键

我有一个 key => value 形式的关联数组,其中 key 是一个数值,但它不是一个连续的数值。键实际上是一个 ID 号,值是一个计数。这对于大多数情况来说都很好,但是我想要一个函数来获取数组的可读名称并将其用作键,而不更改值。

我没有看到执行此操作的函数,但我假设我需要提供旧密钥和新密钥(我都有)并转换数组。有没有一种有效的方法来做到这一点?


K
KernelM
$arr[$newkey] = $arr[$oldkey];
unset($arr[$oldkey]);

请注意 1) 没有两个键具有相同的人类可读版本 2) 没有人类可读版本恰好是数字
此外,这可能会改变数组的顺序,您可能需要小心。甚至 PHP 中的关联数组也是有序的,有时会利用这种顺序。
是的,罗宾很好。有没有办法保持相同的顺序?或者您是否需要创建一个新数组来实现这一目标?
额外问题:如何更改 ID,但保留数组顺序?
如果键值没有改变,您将删除一个数组元素。你可能想检查一下。
D
DiverseAndRemote.com

执行此操作并保留数组顺序的方法是将数组键放入单独的数组中,查找并替换该数组中的键,然后将其与值组合回来。

这是一个可以做到这一点的函数:

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 );
}

谢谢,这真的很有帮助,因为我确实需要保留数组的顺序。在找到此页面之前,我已经尝试过接受的答案。
是的,更喜欢保留数组的顺序,看起来更整洁。
必须保留关键顺序,好一个,就像一个魅力!
如果您想要表演或订单保留,请注意:stackoverflow.com/a/58619985/1617857
这是非常糟糕的实现,性能方面。最好单独保留顺序,或者像本例中那样使用 array_splicestackoverflow.com/questions/3797239/…
R
Ram Sharma

如果您的 array 是根据数据库查询构建的,则可以直接从 mysql 语句更改键:

代替

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

使用类似的东西:

"select ´id´ **as NEWNAME** from ´tablename´..."

与其回避此任务并假设输入数据来自结果集,不如在发布答案之前寻求此澄清。
k
kjg

KernelM 的回答很好,但是为了避免 Greg 在评论中提出的问题(冲突键),使用新数组会更安全

$newarr[$newkey] = $oldarr[$oldkey];
$oldarr=$newarr;
unset($newarr);

这是一个很好的解决方案,只要您的数组大小合理。如果您的数组消耗了超过一半的可用 PHP 内存,这将不起作用。
@kingjeffrey,不是真的。数组值不会被复制,只要它们“只是复制”而不被修改。例如,如果有一个数组包含 10'000 个元素并消耗 40MB 内存,则复制它会消耗存储 10'000 个仅对已存在值的引用而不是值副本的内存所需的内存,因此如果 1 个数组消耗 40MB,则其副本可能会消耗 0.5MB(已测试)。
t
temuri
$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
];
*/

我爱我的数组函数。我正要建议这是一个很好的单行器来重命名所有键并保持数组顺序,但我会推荐你的。
又好又紧。如果您正在处理大型数组并且不想更改所有键,则 map 函数中的行变为 return isset($renameMap[$el]) ? $renameMap[$el] : $el;
这个答案缺少它的教育解释。
这并没有打扰那些投票的人。
T
Tom Ritter

您可以使用第二个关联数组,将人类可读的名称映射到 id。这也将提供多对一的关系。然后做这样的事情:

回声“小部件:”。 $data[$humanreadbleMapping['Widgets']];


s
spreadzz

如果您还希望新数组键的位置与旧数组键的位置相同,您可以这样做:

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;
}

L
Léo Benoist

两种解决方案的简单基准比较。

解决方案 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

p
pajafumo

如果你的数组是递归的,你可以使用这个函数:测试这个数据:

    $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;   
}

M
Manoj Sharma

我喜欢 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' )

M
Manoj Sharma

这是一个辅助函数来实现这一点:

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

用法:

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

重命名成功返回true,否则返回false。


请注意,这会改变数组的顺序(重命名的键的元素将位于数组的末尾,而不是在数组中与原来相同的位置)。此外,我通常不会使用下划线开头的函数名称(传统上用于指定特殊的内部使用函数)。
N
Nadir

简单的东西:

此函数将接受目标 $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_search() 时不明确检查 false 是有问题的。如果搜索到的键是第一个元素怎么办?我担心您正在用这个答案教授不可靠的做法。此外,loosing 拼写为 losing
此外,假设询问者需要翻译其原始数组中的所有键,那么您建议对自定义函数进行循环调用,该函数在没有早期 return/break 的情况下进行循环函数调用。这绝对不会表现得很好。
M
Manoj Sharma

此代码将有助于将旧密钥更改为新密钥

$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");

...但提问者并没有寻求将值作为新键应用的方法。如果这是任务,那么 PHP 已经原生提供了 array_combine($keys_array, $keys_array)。实际上,我是说没有人应该使用您的代码段来娱乐——即使它确实提供了欲望效果。
m
mickmackusa

此页面对所需内容进行了广泛的解释,因为问题正文中没有最小的、可验证的示例。有些答案只是试图解决“标题”而没有理解问题要求。

键实际上是一个 ID 号,值是一个计数。这对于大多数情况来说都很好,但是我想要一个函数来获取数组的可读名称并将其用作键,而不更改值。

PHP 键不能更改,但可以替换——这就是为什么这么多答案建议使用 array_search()(性能相对较差)和 unset()

最终,您希望创建一个新数组,其名称作为与原始计数相关的键。这是通过查找数组最有效地完成的,因为搜索键总是优于搜索值。

代码:(Demo)

$idCounts = [
    3 => 15,
    7 => 12,
    8 => 10,
    9 => 4
];

$idNames = [
    1 => 'Steve',
    2 => 'Georgia',
    3 => 'Elon',
    4 => 'Fiona',
    5 => 'Tim',
    6 => 'Petra',
    7 => 'Quentin',
    8 => 'Raymond',
    9 => 'Barb'
];

$result = [];
foreach ($idCounts as $id => $count) {
    if (isset($idNames[$id])) {
        $result[$idNames[$id]] = $count;
    }
}
var_export($result);

输出:

array (
  'Elon' => 15,
  'Quentin' => 12,
  'Raymond' => 10,
  'Barb' => 4,
)

这种技术保持了原始数组顺序(如果排序很重要),不会进行任何不必要的迭代,并且由于 isset() 将非常迅速。


l
lepe

如果您想一次更换多个键(保留顺序):

/**
 * 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);

array_combine() 不要求其参数是索引数组,因此 array_values() 不是必需的。 3v4l.org/uN1ZF 这个答案依赖于第二个数组已经与第一个数组中的 id 完全对齐的事实。这在现实世界的项目中很少见,对于研究人员来说也不太可能成为可靠的解决方案。
@mickmackusa 我认为你是对的。我想这里的问题是这个问题对我来说不是那么清楚。现在重新阅读这个问题(6 年后),我相信你的答案已经确定了(无论如何......我不再使用 PHP......呵呵)。
K
Kristoffer Bohmann

在使用完整数组时,还有另一种方法可以更改数组元素的键 - 无需更改数组的顺序。它只是将数组复制到一个新数组中。

例如,我正在使用包含索引键和关联键的混合多维数组 - 我想用它们的值替换整数键,而不会破坏顺序。

我通过切换所有数字数组条目的键/值来做到这一点 - 这里:['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
        )
)

这如何回答最初的问题?看起来这种偏差不再尊重所要求的内容。
A
Alekzander

您可以基于 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
    )
)

这个答案(完全忽略了提问者所描述的场景)过度设计了 PHP 已经提供本机函数调用的解决方案。我永远不会在我的任何代码中使用你的脚本。我会使用这个:array_column($arr, null, 'id')。但是,我必须再说一遍,您的回答与这个只处理平面数组的问题无关。
M
MikeyJ

这个基本函数处理交换数组键并将数组保持在原始顺序......

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;
}

然后,您可以循环并用 'z' 交换所有 'a' 键......

$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);
}

那么,您是否真的建议提问者应该循环调用一个自定义函数来进行全数组循环? ...也没有break???这将不是一个有效的解决方案。
G
Grant

此函数将通过结合索引搜索重命名数组键,保持其位置。

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');

对于重新映射数组键的任务,这看起来像是很多繁重的工作。
w
wmmso

这适用于重命名第一个键:

$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);
}

如果值可能包含重复项,则翻转数组不是一个稳定的解决方案。提出的问题指出输入数据包含作为键的 id 和作为值的计数。这些计数可能包含重复项。因此,翻转是一个糟糕的选择。你之前写的 unset&union 硬编码太多了。提问者不太可能对整个数组进行单独/手动编码的元素声明。
K
Kamil Dąbrowski

最好的方法是使用参考,而不是使用 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 行 +++ 等长数据,则清理旧数组。


如果您只是要烘焙一个新数组,我看不到涉及引用的好处。
A
Andrew

一个很容易理解的保存者订购的:

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;
}

v
vardius

您可以编写简单的函数,将回调应用于给定数组的键。类似于 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


提问者无意仅仅在原始键上附加一个字符串。请重新阅读问题要求。输入数组的键必须由另一个数组中的关联值替换。最多,这个答案是另一个问题的正确答案。从表面上看,这个答案是完全错误的。
F
Frank Vu

嗯,我以前没有测试过,但我认为这段代码有效

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);
}

Json编码和解码?这是一个非常糟糕的答案。