我有一个包含键和值的数组。每个值都是整数。我有一个具有相同键的其他数组。如何减去匹配键的所有值?也可能有第二个数组中没有出现的键,但两个数组的长度相同。如果数组2中有一个键不存在于数组1中,则其值应保持不变。如果第一个数组中的某个键不在第二个数组中,则应将其丢弃。我该怎么做?这有什么内置功能吗?
如果我要编写一个脚本,那么就像这样循环:
$arr1 = array('a' => 1, 'b' => 3, 'c' => 10);
$arr2 = array('a' => 2, 'b' => 1, 'c' => 5);
$ret = array();
foreach ($arr1 as $key => $value) {
$ret[$key] = $arr2[$key] - $arr1[$key];
}
print_r($ret);
/*
should be: array('a' => 1, 'b' => -2, 'c' => -5)
*/
我没有在这里添加一个密钥在一个数组中而不在另一个数组中。
答案 0 :(得分:18)
如果您愿意,可以使用数组函数来避免foreach。
向array_map
docs提供的结果将从每个对应的$arr1
中减去每个$arr2
值。不幸的是,array_map
在使用多个输入数组时不会保留您的密钥,因此我们使用array_combine
docs将减去的结果合并回原始密钥的数组中:
$arr1 = array('a' => 1, 'b' => 3, 'c' => 10);
$arr2 = array('a' => 2, 'b' => 1, 'c' => 5);
$subtracted = array_map(function ($x, $y) { return $y-$x; } , $arr1, $arr2);
$result = array_combine(array_keys($arr1), $subtracted);
var_dump($result);
<强>更新强>
与简单的foreach相比,我对数组函数的处理方式感兴趣,因此我使用Xdebug对两者进行了基准测试。这是测试代码:
$arr1 = array('a' => 1, 'b' => 3, 'c' => 10);
$arr2 = array('a' => 2, 'b' => 1, 'c' => 5);
function arrayFunc($arr1, $arr2) {
$subtracted = array_map(function ($x, $y) { return $y-$x; } , $arr1, $arr2);
$result = array_combine(array_keys($arr1), $subtracted);
}
function foreachFunc($arr1, $arr2) {
$ret = array();
foreach ($arr1 as $key => $value) {
$ret[$key] = $arr2[$key] - $arr1[$key];
}
}
for ($i=0;$i<10000;$i++) { arrayFunc($arr1, $arr2); }
for ($i=0;$i<10000;$i++) { foreachFunc($arr1, $arr2); }
事实证明,使用foreach
循环比使用数组函数完成相同任务要快一个数量级。从下面的KCachegrind被调用者图像可以看出,数组函数方法在上面的代码中需要将近80%的处理时间,而foreach函数需要不到5%。
这里有一个教训:有时候,更多的语义数组函数(令人惊讶的是?)在PHP中的老式循环中可能性能较差。当然,您应该始终选择更具可读性/语义性的选项;如果这些微观优化在六个月之后使代码更难以理解,那么这样的微观优化是不合理的。
答案 1 :(得分:1)
foreach ($arr2 as $key => $value) {
if(array_key_exists($key, $arr1) && array_key_exists($key, $arr2))
$ret[$key] = $arr2[$key] - $arr1[$key];
}
答案 2 :(得分:0)
PHP不提供矢量化数学运算。我会坚持你目前使用循环的方法。
首先,我会得到每个数组的数组键集。 (参见array_keys
功能)。然后,他们相交。现在,您将拥有每个阵列共有的键。 (看看array_intersect
函数)。最后,迭代。这是一种可读且简单的方法。
最后,您可以查看一个库,例如Math_Vector:http://pear.php.net/package/Math_Vector