<?php
class BaseClass {
protected $data = array("foo" => 0, "bar" => 0, "baz" => 0);
public function __set($name, $value) {
if( array_key_exists($name, $this->data)){
$func = "set_$name";
return call_user_func_array(array($this,$func),array($value));
}
}
public function __get($name) {
if ( array_key_exists($name, $this->data)){
$func = "get_$name";
return call_user_func_array(array($this,$func),array());
}
}
public function __call($method, $args) {
if (method_exists($this,$method)) {
return call_user_func_array(array($this,$method),$args);
}
if (substr($method, 0, 4) == 'set_'){
$var_name = substr($method, 4);
if (!(array_key_exists($var_name, $this->data))){
return FALSE;
}
$this->data[$var_name] = $args[0];
return TRUE;
}
if (substr($method, 0, 4) == 'get_'){
$var_name = substr($method, 4);
if (!(array_key_exists($var_name, $this->data))){
return FALSE;
}
return $this->data[$var_name];
}
}
}
class SubClass extends BaseClass {
protected $baz_changed = FALSE;
public function set_baz($value) {
if ($value != $this->baz){
print "\n\nthis->data BEFORE SET: ";
print_r($this->data);
print "\n\nthis->baz: ";
print_r($this->baz);
print "\n\nPASSED baz: ";
print_r($value);
$this->baz = $value;
print "\n\nbaz AFTER SET: ";
print_r($this->baz); // appears it was set
print "\n\nDATA ARRAY: ";
print_r($this->data); // but it wasn't ... what gives?
$this->baz_changed = TRUE;
}
}
}
$sc = new SubClass();
$sc->foo = 1;
print "\n\$sc->foo = $sc->foo\n";
$sc->baz = 5;
print "\$sc->baz = $sc->baz\n";
?>
我得到以下结果并不像我预期的那样:
$sc->foo = 1
this->data BEFORE SET: Array (
[foo] => 1
[bar] => 0
[baz] => 0 )
this->baz: 0
PASSED baz: 5
baz AFTER SET: 5
DATA ARRAY: Array (
[foo] => 1
[bar] => 0
[baz] => 0 ) $sc->baz = 5
正如您所看到的,似乎baz已设置但它永远不会在数据数组中设置。任何人都可以解释为什么以及如何解决这个问题?
编辑:修复了结果的格式并为此代码部分添加了更多上下文,因为stackoverflow抱怨我没有足够的。
编辑:刚刚注意到最后它说$ sc-&gt; baz = 5.但是数据数组没有更新。这不是预期的我宁愿更新baz的数据阵列版本。而不是在SubClass中创建的新实例变量。
答案 0 :(得分:2)
您试图以递归方式调用__set
,而PHP明确禁止此操作。
__set
,您致电set_baz
set_baz
,您执行$this->baz = 5
__set
,但PHP会阻止这种情况发生。如果没有,你的程序永远不会结束。如果您已经在__set
内,则无法触发__set
。相反,您正在动态定义名为$this->baz
的新成员变量,就像__set
不存在一样。如果您var_dump
您的对象,您会发现它现在包含$data
和$baz
成员。
在set_baz
内,您需要明确写入$this->data
。您无法写信至$this->baz
。
答案 1 :(得分:0)
它的行为正确,当你执行$sc->baz = 5;
时,它发现set_baz方法就在那里并执行此行$this->baz = $value;
并设置baz属性的值
假设类中的__get和__set是否也在处理类中的$ this-&gt;数据?