php实现空对象模式

Posted 幸运儿

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了php实现空对象模式相关的知识,希望对你有一定的参考价值。

空对象模式(Null Object Pattern):用一个空对象取代 NULL,减少对实例的检查。这样的空对象可以在数据不可用的时候提供默认的行为

(一)为什么需要空对象模式

解决在需要一个对象时返回一个null值,使其调用函数出错的情况

(二)空对象模式UML图

上图是Java的空对象模式UML图,网上很多php设计模式的代码实现都是照着上面这个UML图

实际上PHP在空对象模式的实现上比Java更加简单些。因为PHP有美妙的语法糖,魔术方法__call方法。

我们只要实现空对象的__call方法就可以实现空对象模式,并不需要使用nullobject去继承对应的抽象object。

(三)简单实例

假设现在我们有这么一段代码

<?php
//测试类
class Person{
    public function code(){
        echo ‘code makes me happy‘.PHP_EOL;
    }
}


//定义一个生成对象函数,只有PHPer才允许生成对象
function getPerson($name){
    if($name==‘PHPer‘){
        return new Person;
    }
}

$phper = getPerson(‘PHPer‘);
$phper->code();

是的,现在这段代码会输出code makes me happy。如果有时候这个函数是别人调用的,它并没传入合适的参数呢?

$phper = getPerson(‘Javaer‘);
$phper->code();

这个时候就会报错了error : Call to a member function code() on null。是的$phper现在是一个null值,所以调用code方法就会报错

这种情况很常见,系统并没有返回一个我们期待的对象,而是返回了一个null值。所以多数情况下,我们的代码都要这样写

$phper = getPerson(‘Javaer‘);
if(!is_null($phper)){
    $phper->code();
}

或者是

if(is_object($phper)){
    $phper->code();
}

这样让太多的if判断不可避免地存在于我们的代码中
如果我们使用NullObject模式的话,我们就可以让函数没有返回值时返回一个nullobject对象。而不是一个null值(没有return 默认null值)

//测试类
class Person{
    public function code(){
        echo ‘code makes me happy‘.PHP_EOL;
    }
}


//空对象模式
class NullObject{}

//定义一个生成对象函数,只有PHPer才允许生成对象
function getPerson($name){
    if($name==‘PHPer‘){
        return new Person;
    }else{
        return new NullObject;
    }
}

$phper = getPerson(‘PHer‘);
$phper->code();

这个时候就不会再报一个null调用函数的错误了,但是会报一个call to undefined method的错误,这是由于NullObject对象没有code方法。这个时候我们只需实现魔术方法__call,就不会报错了。

//空对象模式
class NullObject{
    public  function __call($method,$arg){
        echo ‘this is NullObject‘;
    }
}

我们可以通过返回一个NullObject对象来取代返回null,这样就不用在调用方法时判断是否为null,而且只要你实现了call方法,不管真正的对象它原来是调用那个方法的,NullObject都可以去调用而且不报错(实际是隐式调用了魔术方法call)。当然,如果你原本的逻辑是返回对象是null的话什么都不做,那么你可以让__call()什么都不做。或者你也可以让它抛出一个异常。


以上是关于php实现空对象模式的主要内容,如果未能解决你的问题,请参考以下文章

案例分析:设计模式与代码的结构特性

设计模式系列 - 空对象模式

空对象模式

php如何实现多继承?

07_空对象模式

PHP面向对象之选择工厂和更新工厂