【问题标题】:Why array_diff() gives Array to string conversion error?为什么 array_diff() 给出数组到字符串的转换错误?
【发布时间】:2013-11-18 18:46:31
【问题描述】:

以下行出现数组到字符串转换错误:

$diff = array_diff($stockist, $arr);

这里,$arr 是从 JSON 文件解码的数组。使用is_array() 函数,我能够验证这两个参数都是数组。有人能指出我的问题吗

$stockist = array();
while (!feof($file_handle)) {

    $line_of_text = fgetcsv($file_handle);
    $query = "SELECT * FROM reorderchart WHERE medicine = '"
        . trim($line_of_text[3])
        . "' ORDER BY medicine";
    $result = mysql_query($query);

    if (trim($line_of_text[2]) - trim($line_of_text[1]) <= 0) {

        while ($row = mysql_fetch_array($result)) {

            $file = "results.json";
            $arr = json_decode(file_get_contents($file),true);
            $pharmacy = trim($row['Medicine']);

            if (isset($stockist[$pharmacy])) {

                $medicine = $stockist[$pharmacy];
                $medicine[] = trim($row['Stockist']);
                $stockist[$pharmacy] = $medicine;

            } else {

                $medicine = array();
                $medicine[] = trim($row['Stockist']);
                $stockist[$pharmacy] = $medicine;
            }
        }
    }
}
$diff = array();
$diff = array_diff_assoc($stockist,$arr);
ksort($diff);
foreach ($diff as $key => $value) {

    echo "<table align='center' border='1'>";
    echo "<tr><td align = 'center'> <font color = 'blue'> $key</td></tr>";

    foreach($value as $key1 => $value1) {

        echo "<tr><td align ='center'>$value1</td></tr><br>";
    }
    echo "</table>";
}

【问题讨论】:

  • 可能原因:多维数组
  • “两个参数都是数组” 当然是,php 说的没什么不同。问题是数组到字符串的转换,而不是相反。问题是这些数组包含
  • @viakondratiuk 是的,它是多维的。但不是 array_diff() 只评估第一级
  • @user2963765 此函数仅检查 n 维数组的一维。当然你可以使用array_diff($array1[0], $array2[0]);来检查更深的维度。
  • 您可以使用recursive array_diff。看看这个gist.github.com/jondlm/7709e54f84a3f1e1b67b,它解决了我的问题......

标签: php arrays multidimensional-array array-difference


【解决方案1】:

据此:

php -r 'array_diff(array("a" => array("b" => 4)), array(1));'
PHP Notice:  Array to string conversion in Command line code on line 1
PHP Stack trace:
PHP   1. {main}() Command line code:0
PHP   2. array_diff() Command line code:1

您的其中一个数组是多维的。

array_diff 只检查 n 维数组的一维。当然你可以使用array_diff($array1[0], $array2[0]);来检查更深的维度

【讨论】:

  • 对我来说似乎是一个错误。 “数组到字符串的转换”消息充其量是误导性的。
  • 快到 2020 年了,仍然存在的信息误导了我
  • 现在是 2020 年,我同样对这个错误感到困惑。
  • 现在是 2021 年,我讨厌一切。
  • 现在是 2022 年,我仍然讨厌这一切
【解决方案2】:

是的,严格的答案是因为“您的数组之一是多维的。”

另一个有用的注释可能是 - 取决于您进一步解析实际差异的需要 - 考虑首先测试您的数组:

$diff = strcmp(json_encode($stockist), json_encode($arr));

$diff = strspn(json_encode($stockist) ^ json_encode($arr), "\0");

$diff = xdiff_string_diff(json_encode($stockist), json_encode($arr));

所有这些选项都会比较整个数组树,而不仅仅是顶层。

【讨论】:

  • $diff = strcmp(json_encode($stockist), json_encode($arr));不起作用,当两者之间存在差异时总是返回 0
【解决方案3】:

由于array_diff 只能处理一维,您可以:

【讨论】:

  • 多维数组的展平是错误的。当接近第一个嵌套数组时,函数会忘记原始 $arr 中的所有其他值。
  • 完美!与最受欢迎的答案相反,反序列化起作用。只需要做 $results = array_map('unserialize', $results);后记阅读它
  • 第二个例子正是我要找的,谢谢!!
【解决方案4】:

根据函数的 PHP 文档

注意: 当且仅当 (string) $elem1 === (string) $elem2 时,才认为两个元素相等。换句话说:当字符串表示相同时。

更多信息请参考http://php.net/manual/en/function.array-diff.php

【讨论】:

  • 我只需要比较键。所以我使用了array_diff_key()。但它返回一个空数组。
  • 如果它返回一个空数组,这意味着两个数组具有相同的一组键。
【解决方案5】:

你可以在array_diff() documentation看到:

两个元素被认为相等当且仅当 (string) $elem1 === (字符串)$elem2。换句话说:当字符串表示是 一样。

所以看起来你不能将此函数用于多维数组,或者实际上任何不能转换为字符串的值。这是因为该函数会将值转换为 string 以进行比较。

您可以编写自己的函数来递归检查数组是否存在差异 - 事实上,以下内容来自上面链接的文档的 cmets。

你可以看到comment here

function arrayRecursiveDiff($aArray1, $aArray2) { 
    $aReturn = array(); 

    foreach ($aArray1 as $mKey => $mValue) { 
        if (array_key_exists($mKey, $aArray2)) { 
            if (is_array($mValue)) { 
                $aRecursiveDiff = arrayRecursiveDiff($mValue, $aArray2[$mKey]); 
                if (count($aRecursiveDiff)) { $aReturn[$mKey] = $aRecursiveDiff; } 
            } else { 
                if ($mValue != $aArray2[$mKey]) { 
                    $aReturn[$mKey] = $mValue; 
                } 
            } 
        } else { 
            $aReturn[$mKey] = $mValue; 
        } 
    } 

    return $aReturn; 
}

【讨论】:

  • 令人吃惊的是,PHP 语言作者认为不编写函数的递归版本就可以了。并将数组保留为字符串转换错误,而不是让错误说“不支持多维数组”
【解决方案6】:

今天在为 PHP 7.3 从 5.3 进行平台升级时遇到了这个问题……因为我要存储这个值以供以后使用,所以我想确保我没有以“部分”结束序列化数组,可能会破坏下游。

感谢 kenorb 让我走上正确的道路(我赞成你的回答)。以下对我很有效。

代码:

$a1 = [
  'foo' => 'bar',
  'bar' => [
    'test' => 'test'
  ],
  'foobar' => 'fizzbuzz'
];

$a2 = [
  'foobar' => 'fizzbuzz',
  'herp' => [
    'derp' => 'herpderp'
  ]
];

$diff = array_diff(array_map('serialize', $a1), array_map('serialize', $a2));

$multidimensional_diff = array_map('unserialize', $diff);

print_r($multidimensional_diff);

输出:

Array
(
    [foo] => bar
    [bar] => Array
        (
            [test] => test
        )

)

【讨论】:

  • 与kenorb提供的答案有何不同?
  • 它包括使用 array_map('unserialize', $diff) 以便 $diff 数组的格式保持多维而没有任何序列化。我在 cmets 中看到了它,但是那个文本很小而且很容易被忽略。
  • 稍微扩展你的答案:function diffMultiDim($dataLeft, $dataRight) { return array_map('unserialize', array_diff(array_map('serialize', $dataLeft), array_map('serialize', $数据右))); }
【解决方案7】:

这是我对类似问题的解决方案。我想比较两个关联数组并返回更改后的值,但其中一些元素是数组。所以如果我使用

array_diff_assoc

,它给了我“数组到字符串错误”。我的函数还将比较数组元素,如果有差异,它将返回数组元素。它仍在进行中,尚未进行广泛测试。 例如:

public static $example1 = [
    'id' => 1,
    'status' => 2,
    'elements' => ['test', 'example'],
    'different' => ['old' => 5, 'new' => 9]
];

public static $example2 = [
    'id' => 1,
    'status' => 3,
    'elements' => ['test', 'example'],
    'different' => ['old' => 5, 'new' => 8]
];

public static function test(){
    die(var_dump(self::udiffAssoc(self::$example1, self::$example2)));
}

public static function udiffAssoc(array $array1, array $array2)
{
    $checkDiff = function ($a, $b) use (&$checkDiff) {
        if (is_array($a) && is_array($b)) {
            return array_udiff_assoc($a, $b, $checkDiff);
        } elseif (is_array($a)) {
            return 1;
        } elseif (is_array($b)) {
            return -1;
        } elseif ($a === $b) {
            return 0;
        } else {
            return $a > $b ? 1 : -1;
        }
    };
    return array_udiff_assoc($array1, $array2, $checkDiff);
}

如果你运行 ::test 它将返回:

array(2) {
    ["status"]=>
    int(2)
    ["different"]=>
    array(2) {
    ["old"]=>
    int(5)
    ["new"]=>
    int(9)
  }
}

【讨论】:

    【解决方案8】:

    我也遇到了同样的错误,发现下面的 php 错误报告:

    https://bugs.php.net/bug.php?id=60198

    比较多个数组中元素的一些array_*函数 通过 (string)$elem1 === (string)$elem2 这样做。

    如果 $elem1 或 $elem2 是一个数组,那么数组到字符串通知是 扔了。

    可以抛出 this 的两个函数示例是 array_intersect() 和 array_diff()。

    如果不期望这些函数与其他数组一起使用数组 作为值,这应该在文档页面上提及。

    该报告描述了为什么 php 在比较多维数组时会抛出错误。

    【讨论】:

      【解决方案9】:

      我的解决方案呢:

      $diff = strcmp(serialize($arr1), serialize($arr2))
      

      【讨论】:

      • 你的解决方案只返回差异,但我需要返回孔级别和差异的关键
      【解决方案10】:

      我们收到强制转换为字符串警告,因为 array_diff 在比较之前将元素转换为字符串...并且我们已经传递了整数,关键是添加一个包装器对象,因为对象可以转换为字符串,然后比较时完成后,我们将它们返回到整数,如下所示

      或者-

      您可以使用库https://github.com/voku/Arrayy#diffarray-array-static 它的 has 方法称为 diff() 也可以比较纯整数和其他数据类型

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2018-10-15
        • 2022-01-25
        • 2014-01-08
        • 1970-01-01
        • 2015-02-26
        • 2012-07-15
        • 2017-12-02
        相关资源
        最近更新 更多