2

我正在尝试为我正在处理的项目动态创建数据库实体泛化的基础。我基本上想为扩展它的任何类中的属性动态创建一组标准方法和工具。就像您通过 Python/Django 免费获得的工具一样。

我从这个人那里得到了这个想法:http: //www.stubbles.org/archives/65-Extending-objects-with-new-methods-at-runtime.html

所以我已经实现了上面帖子中描述的 __call 函数,

public function __call($method, $args) {
    echo "<br>Calling ".$method;
    if (isset($this->$method) === true) {
        $func = $this->$method;
        $func();
    }
}

我有一个函数,它通过 get_object_vars 为我提供对象公共/受保护的属性,

public function getJsonData() {
    $var = get_object_vars($this);
    foreach($var as &$value) {
        if (is_object($value) && method_exists($value, 'getJsonData')) {
            $value = $value->getJsonData;
        }
    }
    return $var;
}

现在我想为他们创建一些方法:

public function __construct() {
    foreach($this->getJsonData() as $name => $value) {
        // Create standard getter
        $methodName = "get".$name;
        $me = $this;
        $this->$methodName = function() use ($me, $methodName, $name) { 
            echo "<br>".$methodName." is called";
            return $me->$name; 
        };
    }
}

感谢 Louis H. 在下面指出了“使用”关键字。这基本上会即时创建一个匿名函数。该函数是可调用的,但它不再位于其对象的上下文中。它会产生“致命错误:无法访问受保护的财产”

不幸的是,我绑定到 PHP 5.3 版,它排除了 Closure::bind。因此,在 PHP中延迟加载类方法中建议的解决方案在这里不起作用。

我在这里很困惑......还有其他建议吗?

更新

为简洁起见进行了编辑。

4

2 回答 2

2

像这样尝试(你必须使你需要的变量可用于方法)

$this->$methodName = function() use ($this, $methodName, $name){ 
    echo "<br>".$methodName." is called";
    return $this->$$name; 
};

您应该可以通过$this.

于 2013-01-25T13:46:11.330 回答
0

我没有更新上面的原始问题,而是在此处为遇到相同问题的任何人提供了完整的解决方案:

首先,由于闭包不能访问真正的对象,我需要在创建闭包函数时将实际值包含在“use”声明中(参见上面的原始 __construct 函数):

$value =& $this->$name;
$this->$methodName = function() use ($me, $methodName, &$value)  { 
    return $value; 
};

其次,__call 魔术方法不仅需要调用闭包函数,还需要从它返回任何输出。所以我不只是调用 $func(),而是返回 $func();

这成功了!:-)

于 2013-01-29T11:35:59.430 回答