题 动态创建PHP类函数


我想迭代一个数组并根据每个项动态创建函数。我的伪代码:

$array = array('one', 'two', 'three');

foreach ($array as $item) {
    public function $item() {
        return 'Test'.$item;
    }
}

我该怎么做呢?


14
2017-10-12 22:39


起源


我可以问你为什么要创建这个功能 - Baba
PHP不像那样工作。 - hakre
添加太多动态会使程序无法读取 - 这相当于不可维护。你能详细了解你拥有什么以及想要得到什么吗? - Sven
可能重复 在PHP中动态创建实例方法 - Luís Cruz
@Baba和Sven Simple,他提出了一个很好的观点,因为有些函数似乎只有一个单词的差异。我们正在复制代码。所以最好的方法是编写动态代码。 - Abhi


答案:


您可以使用魔术方法而不是“创建”功能 __call(),因此,当您调用“不存在”功能时,您可以处理它并执行正确的操作。

像这样的东西:

class MyClass{
    private $array = array('one', 'two', 'three');

    function __call($func, $params){
        if(in_array($func, $this->array)){
            return 'Test'.$func;
        }
    }
}

然后你可以打电话:

$a = new MyClass;
$a->one(); // Testone
$a->four(); // null

DEMO: http://ideone.com/73mSh

编辑:如果您使用的是PHP 5.3+,那么实际上 能够 在你的问题中做你想做的事!

class MyClass{
    private $array = array('one', 'two', 'three');

    function __construct(){
        foreach ($this->array as $item) {
            $this->$item = function() use($item){
                return 'Test'.$item;
            };
        }
    }
}

这确实有效,除了你不能打电话 $a->one() 直接,你需要 将其保存为变量

$a = new MyClass;
$x = $a->one;
$x() // Testone

DEMO: http://codepad.viper-7.com/ayGsTu


22
2017-10-12 22:44



@NullUserException:感谢您添加的事实 __call() 是一种“神奇的方法”。 - Rocket Hazmat
你也可以使用魔法 __get() 函数调用闭包/回调函数然后:,参见 在PHP中动态创建实例方法  - 如果你真的在想 __call() 要么 __get() 已被要求提出,请将现有问题作为重复提出。 - hakre
你怎么能用PHP docblock这样编辑器不会警告“不存在的功能”? - Kyslik


class MethodTest
{
    private $_methods = array();

    public function __call($name, $arguments)
    {
        if (array_key_exists($name, $this->_methods)) {
            $this->_methods[$name]($arguments);
        }
        else
        {
            $this->_methods[$name] = $arguments[0];
        }
    }
}

$obj = new MethodTest;

$array = array('one', 'two', 'three');

foreach ($array as $item) 
{
    // Dynamic creation
    $obj->$item((function ($a){ echo "Test: ".$a[0]."\n"; }));
    // Calling
    $obj->$item($item);
}

以上示例将输出:

Test: one
Test: two
Test: three

2
2018-06-19 15:45



有没有办法绕过 $a[0] 而且只是 $a ? - duck
@duck class MethodTest {public function __call($ name,$ arguments){echo“”。 “方法:”。$ name。“\ n”。 (!empty($ arguments)?“Parameters:”。implode(',',$ arguments):“没有参数!”)。 “\ n” 个; $ obj = new MethodTest; $ obj-> ExecTest('par 1','par 2','others ......'); $ obj-> ExecTest(); - q81


不确定您的情况下的使用情况,您可以使用 create_function 创建匿名函数。


-2
2017-10-12 22:44



我不认为 create_function 可以用来创建方法,我认为无论如何都要避免它。神奇的方法 __call,正如@RockedHazmat所提出的更好的选择。 - GolezTrol