为了账号安全,请及时绑定邮箱和手机立即绑定

为什么 array_unique 不检测重复对象?

为什么 array_unique 不检测重复对象?

PHP
慕尼黑的夜晚无繁华 2023-04-02 14:46:44
我似乎无法弄清楚 PHP 场景背后发生了什么魔法以及为什么array_unique无法检测到我的重复项。在我的具体情况下,我有 2 个用户集合,我将它们合并为一个,然后只保留唯一的条目。为此,我将两个集合都转换为数组,array_merge()然后基于参数应用array_unique(..., SORT_REGULAR),以便将它们作为对象进行比较而无需任何转换。我意识到比较对象是一个滑坡,但在这种情况下它比我想象的更奇怪。合并之后但在唯一性检查之前我有这个状态:如您所见,第 4 项和第 11 项是相同的用户实体(非严格比较和严格比较都同意这一点)。然而,array_unique()由于某种原因,他们都留在了名单中:如您所见,第 7-10 项已被检测并删除,但第 11 项没有。这怎么可能?我在这里没有看到什么?当前运行 PHP 7.4.5代码来自使用 Symfony 4.4.7 和 Doctrine ORM 2.7.2 的项目(尽管我认为这应该是无关紧要的,如果对象在比较==和===比较中都是相等的)。奖励积分的有趣事实 -array_unique连续申请两次实际上会产生独特的结果:头脑=吹更新:throw new \RuntimeException()我在我的方法中添加了User::__toString(),以额外确保没有人正在转换为字符串。请不要建议转换为字符串 - 这既不是我的问题的解决方案,也不是这个问题的内容。
查看完整描述

3 回答

?
有只小跳蛙

TA贡献1824条经验 获得超8个赞

array_unique对于您手头的问题,我真的怀疑这是从使用标志时从数组中删除元素的方式来的 SORT_REGULAR,方法是:

  1. 整理它

  2. 如果相邻项相等,则删除它们

Proxy而且因为您的收藏中间确实有一个对象User,这可能会导致您遇到当前面临的问题。

这似乎得到了sortPHP 文档页面警告的支持。

警告对具有混合类型值的数组进行排序时要小心,因为如果是sort(),可能会产生意想不到的结果。sort_flagsSORT_REGULAR


现在对于一个可能的解决方案,这可能会给你带来更多 Symfony 风味的东西。

它使用ArrayCollection filterandcontains方法来过滤第二个集合,只添加第一个集合中不存在的元素。
为了完全完整,该解决方案还利用use语言结构将第二个传递ArrayCollection给所需的闭包函数filter

这将导致一个新的ArrayCollection不包含重复的用户。

public static function merge(Collection $a, Collection $b, bool $unique = false): Collection {

  if($unique){

    return new ArrayCollection(

      array_merge(

        $a->toArray(),

        $b->filter(function($item) use ($a){

          return !$a->contains($item);

        })->toArray()

      )

    );

  }


  return new ArrayCollection(array_merge($a->toArray(), $b->toArray()));

}


查看完整回答
反对 回复 2023-04-02
?
炎炎设计

TA贡献1808条经验 获得超4个赞

我知道你说你不想转换为字符串,但我看到你还没有出路,所以我建议你对serialize数组中的每个对象使用该函数,我没有找到方法比较未转换为数组或字符串的对象(如果您不熟悉字符串或数组,则不能尝试转换为二进制或十六进制,但我不知道是否可以转换为二进制或十六进制而不转换为字符串).


但是,如果你使用serialize,你可以在 php 的读取数据中序列化对象,与其他序列化对象相比,这个方法 ( serialize) 是安全的,因为你可以做一个unserialize, 并再次获得原始对象。


所以你可以序列化数组中的所有元素,然后,你可以array_unique像这样使用:


<?php


header("Content-Type: application/json");


class MyClass

{

    public $var1;

    public $var2;

    function __construct($var1, $var2)

    {

        $this->var1 = $var1;

        $this->var2 = $var2;

    }


}


$arr = [

    "a",

    "a",

    [1,2,3],

    "b",

    [1,2,3],

    new MyClass(1,1),

    new MyClass(1,new MyClass(1,1)),

    new MyClass(1,new MyClass(1,1)),

];


$arrSerilized = array_map("serialize", $arr);


var_dump(

    array_map(

        "unserialize",

        array_unique(

            $arrSerilized,

            SORT_STRING

        )

    )

);


/* output:

array(5) {

    [0]=>

    string(1) "a"

    [2]=>

    array(3) {

        [0]=>

        int(1)

        [1]=>

        int(2)

        [2]=>

        int(3)

    }

    [3]=>

    string(1) "b"

    [5]=>

    object(MyClass)#6 (2) {

        ["var1"]=>

        int(1)

        ["var2"]=>

        int(1)

    }

    [6]=>

    object(MyClass)#7 (2) {

        ["var1"]=>

        int(1)

        ["var2"]=>

        object(MyClass)#8 (2) {

            ["var1"]=>

            int(1)

            ["var2"]=>

            int(1)

        }

    }

}

*/

希望这对你有帮助,祝你有美好的一天!


PS:你serialize可以在不同的变量类型中保存相同的值,比如1在"1"不同的php读取数据中序列化


查看完整回答
反对 回复 2023-04-02
?
牛魔王的故事

TA贡献1830条经验 获得超3个赞

在不了解您的实体类的情况下,很难猜测为什么会这样。但我想你的主要问题是__toString()method 。如果您尚未定义它,则应添加一个,以便它为每个实体对象返回一个唯一/不同的字符串。如果它已经定义,请确保它返回不同的字符串。


class User{ 

   private $name;


   function __construct($name){ 

      $this->name=$name;

   }


   function __toString(){ 

     return $this->name; 

   }

}


$user = [];

$users[] = new User("User1");

$users[] = new User("User2");

$users[] = new User("User3");


$user1= $users[0];

$users[]=$user1; //duplicate


echo(count(array_unique($users))); // output should be 3

鉴于有关实体类的信息有限,我可以猜到这里。


编辑:


阅读您的编辑后,我猜您将自己锁定在其中。Sincearray_unique将根据您传递的 sort_flag 尝试将实体对象转换为字符串或数字。更多关于array_unique。因此,您需要实现 __toString() 或添加一些公共属性来定义对象对实体的唯一性,例如


class User{ 

       public $id;

       private $name;


       function __construct($id,$name){

          $this->id=$id;

          $this->name=$name;

       }

}


$user = [];

$users[] = new User(1,"User1");

$users[] = new User(2,"User2");

$users[] = new User(3,"User3");


$user1= $users[0];

$users[]=$user1; //duplicate

echo(count(array_unique($users, SORT_REGULAR))); // output should be 3

请注意公共财产$id和SORT_REGULAR旗帜。


查看完整回答
反对 回复 2023-04-02
  • 3 回答
  • 0 关注
  • 142 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信