在PHP中动态创建实例方法

时间:2010-07-12 19:00:51

标签: php oop dynamic

我希望能够在类的构造函数中动态创建实例方法,如下所示:

class Foo{
   function __construct() {
      $code = 'print hi;';
      $sayHi = create_function( '', $code);
      print "$sayHi"; //prints lambda_2
      print $sayHi(); // prints 'hi'
      $this->sayHi = $sayHi; 
    }
}

$f = new Foo;
$f->sayHi(); //Fatal error: Call to undefined method Foo::sayHi() in /export/home/web/private/htdocs/staff/cohenaa/dev-drupal-2/sites/all/modules/devel/devel.module(1086) : eval()'d code on line 12 

问题似乎是lambda_2函数对象没有在构造函数中绑定到$ this。

感谢任何帮助。

2 个答案:

答案 0 :(得分:18)

您正在将匿名函数分配给属性,但随后尝试使用属性名称调用方法。 PHP无法从属性中自动取消引用匿名函数。以下将起作用

class Foo{

   function __construct() {
      $this->sayHi = create_function( '', 'print "hi";'); 
    }
}

$foo = new Foo;
$fn = $foo->sayHi;
$fn(); // hi

您可以利用魔术__call方法拦截无效方法调用,以查看是否存在包含回调/匿名函数的属性:

class Foo{

   public function __construct()
   {
      $this->sayHi = create_function( '', 'print "hi";'); 
   }
   public function __call($method, $args)
   {
       if(property_exists($this, $method)) {
           if(is_callable($this->$method)) {
               return call_user_func_array($this->$method, $args);
           }
       }
   }
}

$foo = new Foo;
$foo->sayHi(); // hi

从PHP5.3起,您还可以使用

创建Lambdas
$lambda = function() { return TRUE; };

有关详细信息,请参阅PHP manual on Anonymous functions

答案 1 :(得分:3)

您可以使用__call魔术方法来使用运行时实例方法。

class Foo
{
    public function __call($name, $args) 
    {
        if ($name == 'myFunc') {
            // call myFunc
        }
    }
}