比较两个具有匹配键的数组中的数值(使用阈值/公差)


Compare numeric values (using a threshold/tolerance) from two arrays with matching keys

我需要比较数据库值与post值。如果post值(十进制价格)在2美分阈值之内,则认为值相等。结果是具有"真实"差异的数组。数组是一致的:相同数量的值,相同的键。

$db_values =   array( "21" => 10.00, "22" => 20.00, "25" => 3.55);
$post_values = array( "21" => 9.98,  "22" => 20.01, "25" => 2.55 ); 

我正在尝试比较绝对值与我的容差值- epsilon(比较PHP中的小数)和array_udiff:

function epsilon_compare ($v1,$v2)
{
 $epsilon = 0.02;
 $diff = abs($v1 - $v2);
   if ($diff <= $epsilon) 
    { 
            return 0;
      //echo "numbers are equal";
      } else {
         return 1;
             }
    }
 print_r(array_udiff($post_values, $db_values, "epsilon_compare"));

给出正确的结果:Array ([25] => 2.55)

,但当我使用不同的数组,我得到错误的结果,例如:

   $db_values =   array( "21" => 10.00, "22" => 20.00, "25" => 3.55);
   $post_values = array( "21" => 8.00,  "22" => 20.01, "25" => 2.55 );

在这个例子中,它给出了:

   Array ( [21] => 8 [22] => 20.01 [25] => 2.55 ) 

Key[22] => 20.01被列出,但它在阈值内,所以它不应该在结果集中。我想我没有完全理解array_udiff。谢谢。

我认为udiff的功能和你想象的不一样。如果你输出v1和v2,它会给你以下信息:

v1: 20.01 v2: 9.98
v1: 2.55 v2: 20.01
v1: 20 v2: 10
v1: 3.55 v2: 20
v1: 9.98 v2: 10
v1: 9.98 v2: 20.01
v1: 20.01 v2: 20
v1: 20.01 v2: 2.55
v1: 2.55 v2: 3.55 

它做的比较比你想象的要多。

你可能需要一段像这样的代码:

function compare_arrays($array1, $array2){
    $result = array();
    foreach($array1 as $value => $outcome){
        if(epsilon_compare($array1[$value], $array2[$value])){
            $result[$value] = $array2[$value];
        }
    }
    return $result;
}

http://docs.php.net/array_udiff:

用户提供的回调函数用于比较。如果认为第一个参数分别小于、等于或大于第二个参数,则它必须返回小于、等于或大于零的整数。
如果元素不相等,则函数总是返回1,从而导致不同的(错误的)比较决策。

<?php
function epsilon_compare ($v1,$v2)
{
    $epsilon = 0.02;
    $diff = $v1 - $v2;
    return abs($diff)<=$epsilon ? 0 : $diff;
}
$db_values =   array( "21" => 10.00, "22" => 20.00, "25" => 3.55);
$post_values = array( "21" => 8.00,  "22" => 20.01, "25" => 2.55 );
print_r(array_udiff($post_values, $db_values, "epsilon_compare"));

打印

Array
(
    [21] => 8
    [25] => 2.55
)

编辑:using array_udiff_assoc

$db_values = array ( "1" => 7.55, "2" => 5.45, "3" => 5.45, "4" => 64.45, "5" => 54.75, "6" => 30.40, "7" => 56.99, "8" => 10.90, "9" => 60.85, "11" => 3.25, "12" => 13.05, "13" => 5.45, "14" => 8.00, "15" => 5.45, "16" => 13.05, "17" => 4.35 );
$post_values = array ( "1" => 7.55, "2" => 5.45, "3" => 5.45, "4" => 64.45, "5" => 54.75, "6" => 30.40, "7" => 56.99, "8" => 10.90, "9" => 60.85, "11" => 3.25, "12" => 13.05, "13" => 2.45, "14" => 8.00, "15" => 5.45, "16" => 12.05, "17" => 2.34 );
print_r(array_udiff_assoc($post_values, $db_values, "epsilon_compare"));

打印

Array
(
    [13] => 2.45
    [16] => 12.05
    [17] => 2.34
)