`
bardo
  • 浏览: 379124 次
  • 性别: Icon_minigender_1
  • 来自: 上海
博客专栏
D1407912-ab64-3e76-ae37-b31aa4afa398
浅述PHP设计模式
浏览量:11824
9d6df9f7-91da-3787-a37c-0e826525dd5d
Zend Framewor...
浏览量:10132
85b628bd-a2ed-3de2-a4b1-0d34985ae8b6
PHP的IDE(集成开发环...
浏览量:9521
社区版块
存档分类
最新评论

PHP:在对象上动态添加一个新的方法

    博客分类:
  • PHP
阅读更多

有关在一个对象上动态添加方法,如果你来自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/

 

 

 

分享到:
评论

相关推荐

    深入PHP:面向对象、模式与实践(第3版)(源码)

    - **装饰器模式**:掌握动态地给对象添加额外功能的方法,而不会修改其结构。 - **其他模式**:如适配器模式、建造者模式、代理模式等,它们都是软件设计的重要工具。 3. **实践与最佳实践** - **错误处理与异常...

    深入PHP:面向对象、模式与实践(第三版)高清PDF和完整源码下载.rar

    装饰器模式则可以在不改变对象接口的情况下动态地给对象添加新的行为或责任。 **实践与案例** 书中的实践部分不仅讲解了理论知识,还提供了丰富的实际案例,帮助读者理解如何在项目中应用这些理论。这包括如何使用...

    深入PHP:面向对象、模式与实践(第三版)高清PDF和完整源码

    - **装饰器模式**:动态地给对象添加额外功能,而不会影响其他对象。 - **策略模式**:定义一系列可互换的算法,使它们可以在运行时选择。 - **命令模式**:将请求封装为一个对象,以便于使用不同的请求、队列...

    深入PHP:面向对象、模式与实践(第3版)带源码目录

    5. 装饰器模式(Decorator):动态地给对象添加额外的职责,提供比继承更灵活的扩展。 6. 代理模式(Proxy):为其他对象提供一种代理以控制对这个对象的访问。 四、实践应用 除了理论知识,书中还可能包含如何将...

    深入PHP:面向对象、模式与实践(第3版)第5到8章

    单例模式(Singleton Pattern)确保一个类只有一个实例,观察者模式(Observer Pattern)让多个对象可以监听某对象的状态变化,装饰器模式(Decorator Pattern)动态地给对象添加新功能,而适配器模式(Adapter ...

    深入PHP:面向对象、模式与实践(第3版)第9到12章

    例如,装饰器模式用于动态地给对象添加新的行为或责任,而适配器模式则允许不兼容的接口之间进行通信。此外,本章可能还涵盖了代理模式,它为其他对象提供一个代理以控制对这个对象的访问,以及策略模式,允许在运行...

    php简单面向对象的网站例子

    在这个“php简单面向对象的网站例子”中,我们将探讨如何在PHP中实现面向对象的基本原理,包括类、对象、属性、方法以及继承、封装和多态性。 首先,类是面向对象编程的基础,它是创建对象的蓝图。在PHP中,我们...

    一个ASP创建动态对象的工厂类(类似PHP的stdClass)

    例如,`getReadOnlyCode`函数生成了一个私有变量和对应的公共getter方法的代码,这样就可以在对象外部访问这些动态添加的属性。 使用这个工厂类,开发者可以在运行时根据需要动态地创建和扩展对象,从而增加了代码...

    深入PHP++面向对象、模式与实践_2017英文版+最新中文版(附源码)

    5. **装饰器模式**:动态地给对象添加一些额外的职责,为对象添加新的行为或责任。 在实践部分,本书将结合实际项目案例,讲解如何在PHP开发中有效地应用这些理论知识,包括但不限于数据库操作、MVC框架、Web服务等...

    PHP 面向对象 设计模式详解

    6. 装饰器模式:动态地给一个对象添加一些额外的职责,提供了一种用独立于对象实例的附加操作来扩展对象功能的方式。 7. 代理模式:为其他对象提供一种代理以控制对这个对象的访问。 了解并熟练应用这些设计模式,...

    php面向对象程序设计类

    在上面的例子中,`MyClass`是一个类,包含了一个公共属性`$property`和一个公共方法`myMethod()`。 2. **类的实例化** 实例化是创建类的实例(对象)的过程。使用`new`关键字可以创建类的实例: ```php $...

    面向对象动态创建表格(php)

    在IT行业中,面向对象编程(Object-Oriented Programming, OOP)是一种常用的设计模式,它以现实世界中的实体为模型,通过类(Class)和对象(Object)来抽象和模拟真实场景。在PHP中,使用面向对象的方式进行开发...

    PHP面向对象技术(全面讲解)(高洛峰)

    面向对象编程(Object Oriented Programming,简称 OOP)是一种程序设计思想,它的核心在于将程序中的数据和操作这些数据的方法组织在一起,形成一个整体,即对象。在OOP中,程序是由多个对象组成的,每个对象都有...

    PHP面向对象教程

    它可以在对象的方法内部使用,以便访问该对象的属性和其他方法: ```php class MyClass { public $property; public function setProperty($value) { $this->property = $value; // 使用$this访问当前对象的属性...

    php面向对象基础详解【星际争霸游戏案例】

    在星际争霸中,可以定义一个`Attackable`接口,规定所有可以攻击的单位必须实现攻击方法: ```php interface Attackable { function attack($target); } class marine implements Attackable { // 实现attack...

    PHP_面向对象教程(强力推荐)

    - **重载方法**:在同一类中定义多个同名但参数列表不同的方法。 - **访问类型**:控制成员的访问级别。 - **静态成员**:不属于任何对象,可以直接通过类名访问。 - **final关键字**:用于禁止类被继承或方法被覆盖...

    PHP面向对象类的实例-计算器

    在这个“PHP面向对象类的实例-计算器”中,我们将创建一个名为`Calculator`的类,该类将包含四个方法:`add()`, `subtract()`, `multiply()` 和 `divide()`,分别对应加、减、乘、除四种基本运算。以下是这个类的...

    PHP100视频教程22:PHP面向对象开发的学习(六).rar

    **继承**允许我们创建一个新的类,它从已存在的类(父类或基类)继承属性和方法。这样,子类可以复用父类的功能,并添加自己的特定功能。在PHP中,使用`extends`关键字来实现继承,如`class ChildClass extends ...

    PHP读取PPT文件的方法

    10. 注意事项:在使用第三方类库时,需要注意两点:一是了解函数和类的参数含义,二是注意字符集设置,防止数据在处理过程中出现乱码。 11. 类库的使用场景:PHPPowerPoint类库在多种场景下非常实用,例如需要在Web...

    由php对象生成json字符串 把php对象变成json字符串.zip

    为了实现这一点,你可以在你的类中添加以下方法: ```php class Person implements JsonSerializable { private $name; protected $age; public $city; public function __construct($name, $age, $city) { $...

Global site tag (gtag.js) - Google Analytics