PHP:重新排序关联数组

15
在php中,我想要有能力通过将元素移动到数组中的特定位置来重新排序关联数组。不一定需要进行排序,只是我自己选择的重新排序。
例如,假设我有一个如下所示的关联数组:
array(
 'a' => 'Element A',
 'b' => 'Element B',
 'c' => 'Element C',
);

在某些情况下,我可能希望将C移动到B之前并获得以下结果:

array(
 'a' => 'Element A',
 'c' => 'Element C',
 'b' => 'Element B',
);
或者在另一种情况下,我可能想把C移动到A之前,得到以下结果:

或在另一種情況下,我可能想將C移動到A之前,得到以下結果:

array(
 'c' => 'Element C',
 'a' => 'Element A',
 'b' => 'Element B',
);

我想展示的是一种方法,用于表示“嘿,我想将这个数组元素移动到另一个数组元素之前”,或者“嘿,我想移动这个数组元素以确保它在另一个数组元素之后”。

希望这很清楚明白!

提前感谢任何愿意帮助我的人。


1
关联数组没有任何顺序。 - DonCallisto
18
@DonCallisto: 这完全是错误的。数组也是列表,因此绝对有明确定义的顺序。 - Jon
如果您的数组是数值型的,那么使用例如array_splice来实现您想要的操作就会简单得多: http://www.php.net/manual/en/function.array-splice.php - biziclop
3
如果是数字,那么解决方案会很简单 :) 这就是为什么我在Stack Overflow上提问的原因,因为由于关联数组的存在,问题变得很复杂。 - user1494286
7个回答

13
$arr = array(
  'a' => 1,
  'b' => 2,
  'move me' => 9,
  'c' => 3,
  'd' => 4,
);

嘿,我想把['move me']移动到['b']之前。我只需要4行代码就能做到!

$i = 0; foreach($arr as &$val) $val = array('sort' => (++$i * 10), 'val' => $val);
$arr['move me']['sort'] = $arr['b']['sort'] - 5;
uasort($arr, function($a, $b) { return $a['sort'] > $b['sort']; });
foreach($arr as &$val) $val = $val['val'];




我为方便使用编写了一个函数:

function move_item(&$ref_arr, $key1, $move, $key2 = null)
{
  $arr = $ref_arr;
  if($key2 == null) $key2 = $key1;
  if(!isset($arr[$key1]) || !isset($arr[$key2])) return false;

  $i = 0; foreach($arr as &$val) $val = array('sort' => (++$i * 10), 'val' => $val);

  if(is_numeric($move))
  {
    if($move == 0 && $key1 == $key2) return true;
    elseif($move == 0) { $tmp = $arr[$key1]['sort']; $arr[$key1]['sort'] = $arr[$key2]['sort']; $arr[$key2]['sort'] = $tmp; }
    else $arr[$key1]['sort'] = $arr[$key2]['sort'] + ($move * 10 + ($key1 == $key2 ? ($move < 0 ? -5 : 5) : 0));
  }
  else
  {
    switch($move)
    {
      case 'up':     $arr[$key1]['sort'] = $arr[$key2]['sort'] - ($key1 == $key2 ? 15 : 5); break;
      case 'down':   $arr[$key1]['sort'] = $arr[$key2]['sort'] + ($key1 == $key2 ? 15 : 5); break;
      case 'top':    $arr[$key1]['sort'] = 5; break;
      case 'bottom': $arr[$key1]['sort'] = $i * 10 + 5; break;
      default: return false;
    }
  }
  uasort($arr, function($a, $b) { return $a['sort'] > $b['sort']; });
  foreach($arr as &$val) $val = $val['val'];
  $ref_arr = $arr;
  return true;
}


例子:

move_item($arr, 'move me', 'up'); //move it one up
move_item($arr, 'move me', 'down'); //move it one down
move_item($arr, 'move me', 'top'); //move it to top
move_item($arr, 'move me', 'bottom'); //move it to bottom

move_item($arr, 'move me', -1); //move it one up
move_item($arr, 'move me', 1); //move it one down
move_item($arr, 'move me', 2); //move it two down

move_item($arr, 'move me', 'up', 'b'); //move it before ['b']
move_item($arr, 'move me', -1, 'b'); //move it before ['b']
move_item($arr, 'move me', 'down', 'b'); //move it after ['b']
move_item($arr, 'move me', 1, 'b'); //move it after ['b']
move_item($arr, 'move me', 2, 'b'); //move it two positions after ['b']

//Special syntax, to swap two elements:
move_item($arr, 'a', 0, 'd'); //Swap ['a'] with ['d']


我希望这个功能可以帮助很多人,因为它真的很棒!:D


1
我已经添加了这个代码,以便在数组包含空值时也能正常工作 - $arr = array_map(function($v){ return (is_null($v)) ? "" : $v; },$arr); 你认为有更好的方法吗? - iateadonut
这太棒了,只需要把它放入我的项目中,第一次就能成功工作。 - Tyler Collier

11

如果需要自定义排序,例如可以创建一个数组,该数组是键的期望顺序,然后将值与它们相关联。示例:

$input = array("a"=>"Element A","b"=>"Element B","c"=>"Element C");
$order = array("c","a","b");
$out = array();
foreach($order as $k) {
    $out[$k] = $input[$k];
}
< p > $out中的元素将按照指定的顺序排列。 < /p >

这对我的问题不起作用。我需要将一个元素移动到另一个元素之前或之后。因此,我的方法只会给出这两个键。 - user1494286

2

这里有很多复杂的方法 :) 事实上,你可以利用array_slice()的保留键特性。

$new_element = array('new_key' => 'value');

// if needed, find the insertion index by key
$index = array_search('key to search', array_keys($old_array));

// add element at index (note the last array_slice argument)
$new_array = array_slice($old_array, 0, $index+1, true) + $new_element + array_slice($old_array, $index+1, null, true);

2

如果你想交换两个值,可以创建一个如下的函数:

function array_swap($key1, $key2, $array) {
        $newArray = array ();
        foreach ($array as $key => $value) {
            if ($key == $key1) {
                $newArray[$key2] = $array[$key2];
            } elseif ($key == $key2) {
                $newArray[$key1] = $array[$key1];
            } else {
                $newArray[$key] = $value;
            }
        }
        return $newArray;
    }

1
谢谢您的回答,但我需要移动键和值,而不仅仅是值。 - user1494286

1

我最喜欢luky的答案,但它需要指定所有的键。大多数情况下,您只想在数组开头排序一部分键。这个函数会有所帮助:

function reorder_assoc_array(
  $cur,   // current assoc array 
  $order  // array conaining ordered (subset of) keys in $cur
) {
  $result = [];
  // first copy ordered key/values to result array
  foreach($order as $key) {
    $result[$key] = $cur[$key];
    // unset key in original array
    unset($cur[$key]);
  }
  // ... then copy all remaining keys that were not given in $order
  foreach($cur as $key => $value) {
    $result[$key] = $value;
  }
  return $result;
}

例子:

$assoc_arr = [
  'b' => 'bbb',
  'a' => 'aaa',
  'c' => 'ccc',
  'd' => 'ddd'
];

// suppose we want to swap the first two keys and leave the remaining keys as is
$assoc_arr = reorder_assoc_array($assoc_arr, ['a', 'b']);

// ... the order of keys is now a, b, c, d

0

array_splice 不幸地不能与关联数组一起使用,所以这里有一个有点凌乱的替代方案:

$keys = array_keys($arr);
$values = array_values($arr);

$keyIndex = array_search($someKey, $keys);
array_splice($keys, $keyIndex, 1);
array_splice($values, $keyIndex, 1);

$insertIndex = 1;
array_splice($keys, $insertIndex, 0, array($someKey));
array_splice($values, $insertIndex, 0, array($arr[$someKey]));

$arr = array_combine($keys, $values);

0
我根据这里的一个答案编写了一个函数。它接受要排序的关联数组的数组,以及应该重新排序的键数组。
// $data = array of assoc array
// $newKeysOrder = array("c","a","b");
function resort_assoc_array_by_keys($data, $newKeysOrder) {
  foreach($data as $v) {
    $out = [];
    foreach($newKeysOrder as $k) {
      $out[$k] = $v[$k];
    }  
    $new[] = $out;
  }
  return $new;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接