通过引用取消数组元素的设置

12

通过引用方法,我可以访问多维数组内的任何位置,并且可以更改其值。例如:

$conf = array(
    'type' => 'mysql',
    'conf' => array(
            'name' => 'mydatabase',
            'user' => 'root',
            'pass' => '12345',
            'host' => array(
                    '127.0.0.1',
                    '88.67.45.123',
                    '129.34.123.55'
            ),
            'port' => '3306'
    )
);

$value = & $this->getFromArray('type.conf.host');
$value = '-- changed ---';

// result
$conf = array(
    'type' => 'mysql',
    'conf' => array(
            'name' => 'mydatabase',
            'user' => 'root',
            'pass' => '12345',
            'host' => '-- changed ---'
            'port' => '3306'
    )
);

但是,我无法销毁那个部分:

// normally success
unset($conf['type']['conf']['host']);

// fail via reference
$value = & $this->getFromArray('type.conf.host');
unset($value);

有解决方案吗?

4个回答

6

好的,我认为这是更好的答案。要取消设置,您需要获取对容器数组的引用,然后取消设置数组中的元素;

例如:

$value = & $this->getFromArray('type.conf');

unset  $value['host'];

是的 - 如果你引用了数组中的一个值,PHP允许你通过该引用来更新这个值,但是它不允许你通过取消设置该引用来删除数组条目。例如,参见http://3v4l.org/OsYck - Rich

2

引用与硬链接不同。如果您取消设置引用,这将不会取消设置原始值。

<?php 
$a = 5;
xdebug_debug_zval('a'); // a: (refcount=1, is_ref=0), int 5

$b = &$a;
xdebug_debug_zval('a'); // a: (refcount=2, is_ref=1), int 5
xdebug_debug_zval('b'); // b: (refcount=2, is_ref=1), int 5

unset($b);
xdebug_debug_zval('a'); // a: (refcount=1, is_ref=0), int 5

为什么不写一个小的Config类来抽象数据(数组)?由于对象总是按引用传递,因此您不需要自己处理这个问题。
class Config
{
    // ...
}

$config = new Config(array(
    'db' => array(
        'name' => 'mydatabase',
        'user' => 'root',
        'pass' => '12345',
    )
));

$config->get('db.user');
$config->set('db.user', 'newuser');
$config->unset('db.user');
//...

1
引用就像硬链接一样。如果您更改第二个引用,它会更改第一个引用中的值。如果您删除第二个引用,它不会删除第一个引用。这与文件系统中的硬链接非常类似。 - Rich

1
这是我用于取消嵌套键的函数。
public function unsetKey(string $dotSeparatedKey)
{
    $keys = explode('.', $dotSeparatedKey);
    $pointer = &$this->data;
    $current = false; // just to make code sniffer happy
    // we traverse all but the last key
    while (($current = array_shift($keys)) && (count($keys) > 0)) {
        // if some key is missing all the subkeys will be already unset
        if (!array_key_exists($current, $pointer)) {
            // is already unset somewhere along the way
            return;
        }
        // set pointer to new, deeper level
        // called for all but last key
        $pointer = &$pointer[$current];
    }
    // handles empty input string
    if ($current) {
        // we finally unset what we wanted
        unset($pointer[$current]);
    }
}

0

Creating some functions for my framework, think they halp you.

1. Function set value in array using reference for navigation
- if reference ending by name/key, this name/key will be equal setting value
- if reference ending by delimiter, last name/key will be array with setting value

function array_reference_set($input_arr=array(),$reference='',$delimiter='->',$set_var=''){
    switch ($reference){
        case (is_string($reference)):
            $reference = array_reverse(explode($delimiter, $reference),true);
            break;
        case (!is_array($reference)):
            return $input_arr;
    }
    $key = array_pop($reference);
    if (count($reference)<1){
        if($key!=''){
            $input_arr[$key] = $set_var;
        }elseif (!is_array($input_arr) && $key==''){
            $input_arr = array($set_var);
        }elseif ($key==''){
            $input_arr[] = $set_var;
        }
    }else{
        if (!is_array($input_arr)){
            $input_arr = array($key=>array());
        }
        if (isset($input_arr[$key])){
            $input_arr[$key] = $this->array_reference_set($input_arr[$key],$reference,$delimiter,$set_var);
        }else{
            $input_arr[$key] = $this->array_reference_set(array(),$reference,$delimiter,$set_var);
        }
    }
    return $input_arr;
}

$arr = array_reference_set(array(),'a->b->c','->','test');
//equal
$arr = array('a'=>array('b'=>array('c'=>'test')));//or
$arr['a']['b']['c'] = 'test';

$arr = array_reference_set(array(),'a->b->c->','->','test');
//equal
$arr = array('a'=>array('b'=>array('c'=>array('test'))));//or
$arr['a']['b']['c'][] = 'test';

2. Function set unset value from array using reference

- if reference ending is delimiter, then will be unset varible with name/key befor delimiter
- one moment of using this function: you need update array by returned result of function (in the end of code example)

function array_reference_unset($input_arr=array(),$reference='',$delimiter='->'){
    switch ($reference){
        case (is_string($reference)):
            $reference = array_reverse(explode($delimiter, $reference),true);
            break;
        case (!is_array($reference)):
            return $input_arr;
    }
    $key = array_pop($reference);
    if (count($reference)<1 && is_string($key)){
        if ($key!=''){
            unset($input_arr[$key]);
        }else{
            return false;
        }
    }else{
        if (isset($input_arr[$key])){
            $ret = $this->array_reference_unset($input_arr[$key],$reference,$delimiter);
            if ($ret!==false){
                $input_arr[$key] = $ret;
            }else{
                unset ($input_arr[$key]);
            }
        }
    }
    return $input_arr;
}

$arr = array('a'=>array('b'=>array('c'=>'test')));// test subject

$arr = array_reference_unset($arr,'a->b->c','->');//and
$arr = array_reference_unset($arr,'a->b->c->','->');
//equal
unset($arr['a']['b']['c']);

p.s. sorry for my pure English


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接