4

我尝试实现 __isset 魔术方法,例如以下代码,

为什么我总是收到未定义的索引错误?谁能告诉我该怎么做?

class c {
    public $x = array();
    public function __get($name) {
        return $this->x[$name]; //undefined index: #1:a / #2:b / #3:d
    }
    public function __isset($name) {
        return isset($this->x[$name]);
    }
}

$c = new c;
var_dump(isset($c->a));      
var_dump(isset($c->a->b));   #1
var_dump(isset($c->b->c));   #2
var_dump(isset($c->d['e'])); #3

为什么以下代码工作正常?我不明白?

$x = array();
var_dump(isset($x->a->b->c));       
var_dump(isset($x['a']['b']['c']));
4

2 回答 2

2

你执行不力__get

$c = new c();
var_dump(isset($c->a)); // false
var_dump(isset($c->a->b)); // false
var_dump(isset($c->b->c)); // false
var_dump(isset($c->d['e'])); // false


class c {
    public $x = array();

    public function __get($name) {
        return isset($name) ? $this->x[$name] : null;
    }

    public function __isset($name) {
        return isset($this->x[$name]);
    }
}
于 2013-05-02T22:20:21.943 回答
1

您可能期望,PHP 引擎将__isset()在每次访问 PHP 中的隐藏属性之前调用。但那不是真的。检查文档

__isset() 通过在不可访问的属性上调用 isset() 或 empty() 来触发。

因此,这是预期的行为,仅:

var_dump(isset($c->a));

会触发__isset()

所有其他行都会触发__get()。由于您没有设置这些索引,因此这是预期的行为。

将您的代码更改为:

class c {
    public $x = array();
    public function __get($name) {
        var_dump(__METHOD__);
        return $this->x[$name];
    }
    public function __isset($name) {
        var_dump(__METHOD__);
        return isset($this->x[$name]);
    }
}

查看实际调用了哪些方法。这会给你:

c::__isset   <------ called!
bool(false)
c::__get

Notice: Undefined index: a in /tmp/a.php on line 6

Call Stack:
    0.0002     647216   1. {main}() /tmp/a.php:0
    0.0003     648472   2. c->__get() /tmp/a.php:16

<---------------------- not called 
bool(false)
c::__get

Notice: Undefined index: b in /tmp/a.php on line 6

Call Stack:
    0.0002     647216   1. {main}() /tmp/a.php:0
    0.0005     648656   2. c->__get() /tmp/a.php:17

<---------------------- not called     
bool(false)
c::__get

Notice: Undefined index: d in /tmp/a.php on line 6

Call Stack:
    0.0002     647216   1. {main}() /tmp/a.php:0
    0.0006     648840   2. c->__get() /tmp/a.php:18

<---------------------- not called     
bool(false)
于 2013-05-02T22:13:47.730 回答