PHP:在对象上动态添加一个新的方法
php  /  管理员 发布于 4年前   304
有关在一个对象上动态添加方法,如果你来自Ruby语言或您熟悉这门语言,你已经知道它是什么...... Ruby提供给你一种方式来获得一个instancied对象,并给这个对象添加一个额外的方法。
好!不说Ruby了,让我们来谈谈PHP
PHP未提供一个“标准的方式”做这样的事情,这也是没有核心的一部分...
但无论如何,它并没有说我们不能做这样的事情。
因此,让我们看一下代码,我将展示两个实现:一是使用PHP 5.3,另一个使用PHP5.4,这些例子充分利用在PHP 5.3版本中增加的匿名函数(闭包),并且也利用闭包类的优势以及在5.4版本添加的方法绑定。
在PHP 5.3中添加方法的对象
/** * Example in PHP 5.3 */class Meta{ private $methods = array(); public function addMethod($methodName, $methodCallable) { if (!is_callable($methodCallable)) { throw new InvalidArgumentException('Second param must be callable'); } $this->methods[$methodName] = $methodCallable; } public function __call($methodName, array $args) { if (isset($this->methods[$methodName])) { array_unshift($args, $this); return call_user_func_array($this->methods[$methodName], $args); } throw RunTimeException('There is no method with the given name to call'); }}
/** * Example in PHP 5.3 */require 'Meta.php';$meta = new Meta();$meta->addMethod('color', function ($self) { $self->name = 'My Name'; return '#00000';});echo $meta->color(), PHP_EOL;echo $meta->name, PHP_EOL;
GitHub 的链接 https://gist.github.com/krolow/4189729#file-meta-php
它是如何工作的?
它使匿名函数和魔术方法__call,该类元有方法addMethod的使用,这种方法是等待两个参数,第一个是方法名,第二个可调用匿名函数。
每次调用原始对象没有(未在类中声明)的方法,该方法调用一下,函数调用将寻找你是否有给定名称注册一个新的方法,如果是,它调用匿名函数,传参数给当前对象的匿名函数,所以你可以访问类的方法和属性。
它有一定的局限性,你将不能访问私有的方法和属性,但它有可能使用象反射这样的方法来处理。
在PHP5.4中添加方法的对象
trait MetaTrait{ private $methods = array(); public function addMethod($methodName, $methodCallable) { if (!is_callable($methodCallable)) { throw new InvalidArgumentException('Second param must be callable'); } $this->methods[$methodName] = Closure::bind($methodCallable, $this, get_class()); } public function __call($methodName, array $args) { if (isset($this->methods[$methodName])) { return call_user_func_array($this->methods[$methodName], $args); } throw new RunTimeException('There is no method with the given name to call'); } }
require 'MetaTrait.php';class HackThursday { use MetaTrait; private $dayOfWeek = 'Thursday';}$test = new HackThursday();$test->addMethod('when', function () { return $this->dayOfWeek;});echo $test->when();
GitHub中的链接:https://gist.github.com/krolow/4264062#file-test-php
正如你所看到的第二个例子让使用trait,PHP5.4的另一项新功能,这样一来我们使模向合成,这是一个好主意,更解耦你的代码。
trait作为与PHP 5.3的类Meta.php颇为相似,也是对象的用法,这里的区别是,当添加方法时你匿名函数不再需要接收的对象实例作为参数,正如你在测试的例子可以看到的,并且你能够直接访问$this引用的对象的方法和属性,这就是可能的,因为有Closure::bind,,因为有这一招,我们能够注入对象,将通过传参添加方法的匿名函数放到内部范围。所以一旦你调用该方法,它是动态创建的,我们调用的闭包是在你添加方法时存储在内存中的,并且闭包是与实例在同一范围的。所以我们现在能够访private属性, protected等...并它看起来也更好一点,因为我们并不需要强制的第一个参数是对象的实例。
我到底为什么要用它呢?
这只是动态添加方法的一种实践,当你想创造更通用的东西,或为你代码的API创建一个漂亮的接口,它给出了一些灵活性。但是必须注意很重要的一点,也许坏的程序员用这样类似的方法可以做出让人发疯的东西。
原文网址:
http://cobaia.net/php/2012/12/12/php-adding-a-new-method-to-an-object-on-the-fly/
123 在
Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..原梓番博客 在
在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..博主 在
佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..1111 在
佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..路人 在
php中使用hyperf框架调用讯飞星火大模型实现国内版chatgpt功能示例中评论 教程很详细,如果加个前端chatgpt对话页面就完美了..Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号