3

问题:

当手动设置或读取值时,ArrayObject 按预期工作,但是当使用函数(例如 foreach)对其进行迭代时,事情变得很糟糕。

它不调用我定义的 offset* 方法,而是使用 ArrayIterator 方法。

代码:

班上:

class obsecureArray extends ArrayObject {


    public function offsetSet($name, $value) {
        call_user_func_array(array('parent', __FUNCTION__), array(base64_encode($name), base64_encode($value)) );
    }

    public function offsetGet($name) {
        return base64_decode( call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name) ) );
    }

    public function offsetExists($name) {
        return call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name) );
    }

    public function offsetUnset($name) {
        return call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name) );
    }
}

使用示例:

$a = new obsecureArray();
$a['test'] = 'Value';
$a[2] = '1';

define('BR','<br />');
echo 'test: ',$a['test'],BR,'2: ',$a[2],BR;

foreach($a as $key => $value)
    echo 'Key: ',$key,' Value:',$value,BR;

输出:

test: Value
2: 1
Key: dGVzdA== Value:VmFsdWU=
Key: Mg== Value:MQ==

键盘示例。

4

4 回答 4

3

我查看了ArrayObject __construct方法文档,发现与我的问题相关的第三个参数:

迭代器类:

指定将用于 ArrayObject 对象迭代的类。ArrayIterator 是使用的默认类。

ArrayIterator现在我可以用我自己的方法扩展 anoffset*并将它传递给我的构造函数ArrayObject,所以我调查了一下ArrayIterator,它几乎与 相同,ArrayObject只是它显然没有使用外部迭代器。所以我所要做的就是扩展ArrayIterator。并且还必须覆盖currentandkey方法。

所以我有:

class obsecureArray extends ArrayIterator {

    public function offsetSet($name, $value) {
        call_user_func_array(array('parent', __FUNCTION__), array(base64_encode($name), base64_encode($value)));
    }

    public function offsetGet($name) {
        return base64_decode(call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name)));
    }

    public function offsetExists($name) {
        return call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name));
    }

    public function offsetUnset($name) {
        return call_user_func_array(array('parent', __FUNCTION__), (array) base64_encode($name));
    }

    public function key() {
        return base64_decode(parent::key());
    }

    public function current() {
        return base64_decode(parent::current());
    }



}

它可以按预期完美运行。

并为:

$a = new obsecureArray();
$a['test'] = 'Value';
$a[2] = '1';

define('BR','<br />');
echo 'test: ',$a['test'],BR,'2: ',$a[2],BR;

foreach($a as $key => $value)
    echo 'Key: ',$key,' Value:',$value,BR;

我有:

测试:值
2:1
键:测试值:值
键:2 值:1

键盘示例

于 2012-01-11T05:08:11.877 回答
1

您需要实现IteratorIteratorAggregate。您可以更好地控制前者,但后者更易于实现(可能需要更多开销):

class People implements IteratorAggregate {
    protected $children;

    // ...

    public function getIterator() {
        return new ArrayIterator( $this->children );
    }
}

有关该接口的样板方法,请参见迭代器。

于 2012-01-11T04:58:15.130 回答
0

您需要设置一个自定义迭代器类:

<?php

class SecureArray extends ArrayObject {

    public function __construct($array = array()) {
        parent::__construct($array);
        $this->setIteratorClass('SecureIterator');
    }

    public function offsetSet($name, $value) {
        parent::offsetSet( base64_encode($name), base64_encode($value) );
    }

    public function offsetGet($name) {
        return base64_decode( parent::offsetGet( base64_encode($name) ) );
    }

    public function offsetExists($name) {
        return parent::offsetExists( base64_encode($name) );
    }

    public function offsetUnset($name) {
        return parent::offsetUnset( base64_encode($name) );
    }
}

class SecureIterator extends ArrayIterator {

    function current() {
        return base64_decode(parent::current());
    }

    function key() {
        return base64_decode(parent::key());
    }
}
于 2012-01-11T05:07:54.633 回答
0

在我看来,如果您想确保正确处理您的值使用正确的方法,您的代码可以按预期工作:

$a = new obsecureArray();
$a->offsetSet('test', 'Value');
$a->offsetSet(2, '1');

也代替

define('BR','
');

有预定义的 php 属性PHP_EOL

于 2012-01-11T04:55:36.620 回答