PHP运行时类修改

时间:2009-10-20 09:35:43

标签: php oop reflection class dynamic

所以我希望能够在运行时添加/删除类方法。在你告诉我oop中可怕的实践之前,它可能是,但我并不在乎。我希望能够做到这一点的原因是因为我希望应用程序非常模块化,所以一些插件可以扩展一些基类并为其添加方法而不会杀死主应用程序。

例如,假设我有以下课程:

class User {
    protected $Id;
    protected $Name;
    protected $Password;
    protected $PostsPerPage;
 }

并说,一些插件增加了用户更改其可见性设置的可能性,为该类添加了一个$ Visible属性。这堂课将成为:

class User {
    protected $Id;
    protected $Name;
    protected $Password;
    protected $PostsPerPage;
    protected $Visible;
 }

这可以通过__get和__set来实现,但最常见的实现是获取和设置运行时生成的属性,为每个添加的属性编写伪getter和setter是一件麻烦事,并且使用__set是在我看来是禁止的。

我的另一个想法是将用户设置单独存储在另一个数组中,例如$ UserVisiblitySettings [userid],但这使得事情不像我希望的那样OO。

接下来的想法是创建一个帮助类,如下所示:

class UserHelper {
    public function SetVisiblity($user_object,value);
}

然后,我可以使用__get和__set来实现“朋友”方法/类,但这听起来太过于苛刻。如果我这样走,也可以重载__call,__ get和__set。此外,我不太确定这是一个好的OOP实践,它看起来也很难看。

我的最后一个想法是通过使用eval()在运行时动态创建类的一些功能(这是我可以提出的唯一有效的eval使用)。基本上,从文件中获取类定义,执行一些简单的括号查找以找到类的开始和结束括号并将其发送到eval。

使用runkit是不可能的,因为它已经过时了,我宁愿不强迫用户安装一些扩展。

当我查看我的想法时,最简单的一个和更少的cpu密集似乎是重载_call并添加方法在类中注册。

请注意任何不“不要这样做”的想法。

3 个答案:

答案 0 :(得分:7)

RunKit分机可以做到(runkit_method_add()等)

然而,这是一个实验性延伸,你已经瞄准了你的脚......

您还有其他选择:

  • 使用__get()__call()
  • 模拟新字段和方法
  • 使用子类化和工厂模式(class Plugin extends BaseImplementation并让工厂实例化Plugin而不是BaseImplementation)。 Zend Plugin Loader就是这样的。
    它是具有最小开销的解决方案,但也仅限于一个扩展一个类的插件。
  • 添加挂钩并使用委派(策略模式)($this->plugin->onFoo())。 There's library for this

答案 1 :(得分:3)

PHP不允许这样做。它可能是其他方面的动态语言,但是类系统是故意限制的。您可以安装runkit扩展名,这会更改语言以允许在运行时使用类进行模拟(但之后您不再使用普通的PHP),或者您可以使用magic-methods来模拟它

答案 2 :(得分:1)

您可以覆盖该课程,但我不知道您是否可以在同一请求中重新加载该课程。 使用中介设计模式(symfony事件调度程序)可以实现某些行为更改,但您需要提前知道扩展点,并在将来触发扩展类捕获的事件/消息。

如果您可以等待下一个请求,并清除缓存(如果有)。我做了一个可以帮助你的工具。 SourceEditor

这里也有类似问题的更多答案,我把代码示例放在哪里。 How to generate or modify a PHP class at runtime?