我试图将表单中的日期过滤为仅用户更改的内容,并开始使用
array_filter
,因为它似乎完全符合我的要求。我测试了一些表格并遇到了这种意想不到的行为。当“新”值为 1 时,array_diff
不会检测到它。在 3v4l.org 上运行时,另一个意想不到的是 foreach 循环在返回预期结果时实际上比 array_filter
更快。我已经阅读了函数的手册页,并了解它进行了字符串比较,但所有数组值都是字符串开头,所以我不认为这是类型转换问题。
我已经解决了我的最初问题,并且很乐意使用更快的 foreach 循环,但我很感兴趣是否有人可以解释为什么这样工作。
<?php
$array_1 = [
'id' => '42',
'base_id' => '23',
'role_id' => '1',
'title' => 'Manage Account',
'slug' => 'manage_account',
'parent' => '31',
'order' => '1',
'visibility' => '1'
];
$array_2 = [
'id' => '42',
'base_id' => '23',
'role_id' => '99999',
'title' => 'Manage Account',
'slug' => 'manage_account',
'parent' => '31',
'order' => '1',
'visibility' => '1'
];
var_dump(array_diff($array_1, $array_2));
// Result (unexpected)
// array (size=0)
// empty
$diff = [];
foreach ($array_1 as $key => $value) {
if ((string) $array_1[$key] !== (string) $array_2[$key]) {
$diff[$key] = $value;
}
}
var_dump($diff);
// Result (expected)
// array (size=1)
// 'role_id' => string '1' (length=1)
array_diff()
在数组 2 中查找数组 1 的每个值的精确重复项,忽略键。
1
在数组 2 中有重复项,例如在 order
键下。这就是为什么它没有被列为差异。
这种行为是否是最佳的或明显的还是有争议的,但这就是它的工作原理。
如果将
1
更改为 3
,则会报告,因为数组 2 不包含值 3
:
$array_1 = [
'id' => '42',
'base_id' => '23',
'role_id' => '3',
'title' => 'Manage Account',
'slug' => 'manage_account',
'parent' => '31',
'order' => '1',
'visibility' => '1'
];
$array_2 = [
'id' => '42',
'base_id' => '23',
'role_id' => '99999',
'title' => 'Manage Account',
'slug' => 'manage_account',
'parent' => '31',
'order' => '1',
'visibility' => '1'
];
var_dump(array_diff($array_1, $array_2));
// Result (unexpected)
// array (size=1)
// 'role_id' => string '3' (length=1)
如果您希望考虑按键,请使用
array_diff_assoc()
代替。