3

我有一个类,它基本上是一个数组的包装器并实现了 IteratorAggregate。当创建该类的新对象时,它将其值存储在一个名为 $value 的受保护变量中。该值可以是字符串、整数等,也可以是可遍历的对象(如数组)。

这个对象是“递归的”,因为当一个可遍历对象(如数组)被传递给构造函数时,他会创建该类的另一个实例。这是要澄清的构造函数:

class ListItem implements \IteratorAggregate
{
    protected $readOnly = false;
    protected $key;
    protected $value;

    public function __construct($key, $value, $readOnly = false)
    {
        if($readOnly) $this->readOnly = true;
        if(is_numeric($key)) $key = 'index'.$key;
        $this->key = $key;

        if (is_array($value) || $value instanceof Traversable) {
            $this->value = array();
            foreach($value as $k => $v) {
                if(is_numeric($k)) $k = 'index'.$k;
                $this->value[$k] =  new ListItem($k, $v, $readOnly);
            }
        } else {
            $this->value = $value;
        }
    }

    public function __toString()
    {
        if ( is_array($this->value) ) {
            return 'ListItem OBJECT(' . count($this->value) . ')';
        } else {
            return $this->value;
        }
    }

现在,我正在尝试为该类编写一个简单的排序方法。

按 Key 排序,这就像一个魅力:

$success = ksort($this->value, $comparison);

但要按值排序,asort 不起作用,因为我尝试排序的实际值存储在 value 属性中。

所以我尝试使用uasort,如下所示:

   $success = uasort($this->value, function ($a, $b)
   {                    
      if ($a->value == $b->value) return 0;
      else if($a->value < $b->value) return -1;
      else return 1;
   });

但由于某些不清楚的原因,我收到以下错误:

警告:uasort() [function.uasort]: 数组被用户比较函数修改在 /* /* /* /ListItem.php 的第 129 行


问:如果我只是访问 $value 进行比较而不实际更改任何内容,为什么会发生这种情况?

4

1 回答 1

1

似乎闭包(或匿名函数)在全局范围内,这意味着uasort无法访问对象的私有或受保护成员ListItem(尽管其他ListItem对象可以访问其兄弟对象的私有/受保护属性)

这解决了问题:(转换为字符串)

$success = uasort($this->value, function ($objA, $objB) use ($comparison)
{
    $a = (string) $objA;
    $b = (string) $objB;
    if($comparison == ListItem::SORT_NUMERIC) {
        if (is_numeric($a)) $a = (int) $a;
        if (is_numeric($b)) $b = (int) $b;
    }                   
    if ($a == $b) return 0;
    else if($a < $b) return -1;
    else return 1;
});
于 2012-10-20T21:40:21.040 回答