如何使用ArrayObject取消设置嵌套数组?

mpe*_*pen 8 php arrayobject

ideone

示例代码:

<?php
$a = new ArrayObject();
$a['b'] = array('c'=>array('d'));
print_r($a);
unset($a['b']['c']);
print_r($a);
Run Code Online (Sandbox Code Playgroud)

产量

ArrayObject Object
(
    [b] => Array
        (
            [c] => Array
                (
                    [0] => d
                )
        )
)
ArrayObject Object
(
    [b] => Array
        (
            [c] => Array
                (
                    [0] => d
                )
        )
)
Run Code Online (Sandbox Code Playgroud)

您注意到$a['b']['c']即使在取消设置后仍然存在.我希望$a只剩下一个值(b).

在我的实际应用中,我收到以下警告:

间接修改MyClass的重载元素没有任何效果

在哪里MyClass延伸ArrayObject.我有很多代码依赖于能够取消设置这样的嵌套元素,所以我怎样才能使它工作?

qui*_*tin 11

一种方法

<?php
$a      = new ArrayObject();
$a['b'] = array('c' => array('d'));
$d      =& $a['b'];

unset($d['c']);
print_r($a['b']);
Run Code Online (Sandbox Code Playgroud)

打印:

Array
(
)
Run Code Online (Sandbox Code Playgroud)

为了解释为什么你最初使用的语法不会删除元素,需要考虑一点时间.

编辑:行为解释

发生的事情就是将呼叫unset($a['b']['c']);翻译成:

$temp = $a->offsetGet('b');
unset($temp['c']);
Run Code Online (Sandbox Code Playgroud)

因为$temp是副本$a,而不是对它的引用,PHP使用写入时复制内部和创建第二阵列,其中$temp没有['b']['c'],但$a仍然没有.

另一个编辑:可重复使用的代码

所以,无论你采用哪种方式切片,似乎function offsetGet($index)都要试图超负荷function &offsetGet($index)导致麻烦; 所以这里我来瓦特最短的辅助方法,/可以将其添加为的子类的静态或实例方法ArrayObject,不管你的船浮筒:

function unsetNested(ArrayObject $oArrayObject, $sIndex, $sNestedIndex)
{
    if(!$oArrayObject->offSetExists($sIndex))
        return;

    $aValue =& $oArrayObject[$sIndex];

    if(!array_key_exists($sNestedIndex, $aValue))
        return;

    unset($aValue[$sNestedIndex]);
}
Run Code Online (Sandbox Code Playgroud)

所以原始代码就会变成

$a      = new ArrayObject();
$a['b'] = array('c' => array('d'));

// instead of unset($a['b']['c']);
unsetNested($a, 'b', 'c');
print_r($a['b']);
Run Code Online (Sandbox Code Playgroud)

另一个编辑:OO解决方案

好的 - 所以我今天早上一直在争抢b/c我在我的代码中发现了一个错误,并且在修改后,我们可以基于OO实现一个解决方案.

只是你知道我试过了,扩展段错误..:

/// XXX This does not work, posted for illustration only
class BadMoxuneArrayObject extends ArrayObject
{
    public function &offsetGet($index)
    {   
        $var =& $this[$index];
        return $var;
    }   
}
Run Code Online (Sandbox Code Playgroud)

另一方面,实现装饰器就像一个魅力:

class MoxuneArrayObject implements IteratorAggregate, ArrayAccess, Serializable, Countable
{
    private $_oArrayObject;  // Decorated ArrayObject instance

    public function __construct($mInput=null, $iFlags=0, $sIteratorClass='')
    {
        if($mInput === null)
            $mInput = array();

        if($sIteratorClass === '')
            $this->_oArrayObject = new ArrayObject($mInput, $iFlags);
        else
            $this->_oArrayObject = new ArrayObject($mInput, $iFlags, $sIteratorClass);
    } 

    // -----------------------------------------
    // override offsetGet to return by reference
    // -----------------------------------------
    public function &offsetGet($index)
    {
        $var =& $this->_oArrayObject[$index];
        return $var;
    }

    // ------------------------------------------------------------
    // everything else is passed through to the wrapped ArrayObject
    // ------------------------------------------------------------
    public function append($value)
    {
        return $this->_oArrayObject->append($value);
    }

    public function asort()
    {
        return $this->_oArrayObject->asort();
    }

    public function count()
    {
        return $this->_oArrayObject->count();
    }

    public function exchangeArray($mInput)
    {
        return $this->_oArrayObject->exchangeArray($mInput);
    }

    public function getArrayCopy()
    {
        return $this->_oArrayObject->getArrayCopy();
    }

    public function getFlags()
    {
        return $this->_oArrayObject->getFlags();
    }

    public function getIterator()
    {
        return $this->_oArrayObject->getIterator();
    }

    public function getIteratorClass()
    {
        return $this->_oArrayObject->getIteratorClass();
    }

    public function ksort()
    {
        return $this->_oArrayObject->ksort();
    }

    public function natcassesort()
    {
        return $this->_oArrayObject->natcassesort();
    }

    public function offsetExists($index)
    {
        return $this->_oArrayObject->offsetExists($index);
    }

    public function offsetSet($index, $value)
    {
        return $this->_oArrayObject->offsetSet($index, $value);
    }

    public function offsetUnset($index)
    {
        return $this->_oArrayObject->offsetUnset($index);
    }

    public function serialize()
    {
        return $this->_oArrayObject->serialize();
    }

    public function setFlags($iFlags)
    {
        return $this->_oArrayObject->setFlags($iFlags);
    }

    public function setIteratorClass($iterator_class)
    {
        return $this->_oArrayObject->setIteratorClass($iterator_class);
    }

    public function uasort($cmp_function)
    {
        return $this->_oArrayObject->uasort($cmp_function);
    }

    public function uksort($cmp_function)
    {
        return $this->_oArrayObject->uksort($cmp_function);
    }

    public function unserialize($serialized)
    {
        return $this->_oArrayObject->unserialize($serialized);
    }
}
Run Code Online (Sandbox Code Playgroud)

现在这段代码按预期工作:

$a      = new MoxuneArrayObject();
$a['b'] = array('c' => array('d'));
unset($a['b']['c']);
var_dump($a);
Run Code Online (Sandbox Code Playgroud)

仍然需要修改一些代码..; 我没有看到任何方式.