PHP 设计模式系列 —— 空对象模式(Null Object)

1、模式定义

空对象模式并不是 GoF 那本《设计模式》中提到的 23 种经典设计模式之一,但却是一个经常出现以致我们不能忽略的模式。该模式有以下优点:

  • 简化客户端代码
  • 减少空指针异常风险
  • 更少的条件控制语句以减少测试用例

在空对象模式中,以前返回对象或 null 的方法现在返回对象或空对象 NullObject,这样会减少代码中的条件判断,比如之前调用返回对象方法要这么写:

if (!is_null($obj)) { 
    $obj->callSomething(); 
}

现在因为即使对象为空也会返回空对象,所以可以直接这样调用返回对象上的方法:

$obj->callSomething();

从而消除客户端的检查代码。

当然,你可能已经意识到了,要实现这种调用的前提是返回对象和空对象需要实现同一个接口,具备一致的代码结构。

2、UML类图

Null-Object-Design-Pattern-Uml

3、示例代码

Service.php

<?php

namespace DesignPatterns\Behavioral\NullObject;

/**
 * Service 是使用 logger 的模拟服务
 */
class Service
{
    /**
     * @var LoggerInterface
     */
    protected $logger;

    /**
     * 我们在构造函数中注入logger
     *
     * @param LoggerInterface $log
     */
    public function __construct(LoggerInterface $log)
    {
        $this->logger = $log;
    }

    /**
     * do something ...
     */
    public function doSomething()
    {
        // 在空对象模式中不再需要这样判断 "if (!is_null($this->logger))..."
        $this->logger->log('We are in ' . __METHOD__);
        // something to do...
    }
}

LoggerInterface.php

<?php

namespace DesignPatterns\Behavioral\NullObject;

/**
 * LoggerInterface 是 logger 接口
 *
 * 核心特性: NullLogger必须和其它Logger一样实现这个接口
 */
interface LoggerInterface
{
    /**
     * @param string $str
     *
     * @return mixed
     */
    public function log($str);
}

PrintLogger.php

<?php

namespace DesignPatterns\Behavioral\NullObject;

/**
 * PrintLogger是用于打印Logger实体到标准输出的Logger
 */
class PrintLogger implements LoggerInterface
{
    /**
     * @param string $str
     */
    public function log($str)
    {
        echo $str;
    }
}

NullLogger.php

<?php

namespace DesignPatterns\Behavioral\NullObject;

/**
 * 核心特性 : 必须实现LoggerInterface接口
 */
class NullLogger implements LoggerInterface
{
    /**
     * {@inheritdoc}
     */
    public function log($str)
    {
        // do nothing
    }
}

4、测试代码

Tests/LoggerTest.php

<?php

namespace DesignPatterns\Behavioral\NullObject\Tests;

use DesignPatterns\Behavioral\NullObject\NullLogger;
use DesignPatterns\Behavioral\NullObject\Service;
use DesignPatterns\Behavioral\NullObject\PrintLogger;

/**
 * LoggerTest 用于测试不同的Logger
 */
class LoggerTest extends \PHPUnit_Framework_TestCase
{

    public function testNullObject()
    {
        $service = new Service(new NullLogger());
        $this->expectOutputString(null);  // 没有输出
        $service->doSomething();
    }

    public function testStandardLogger()
    {
        $service = new Service(new PrintLogger());
        $this->expectOutputString('We are in DesignPatterns\Behavioral\NullObject\Service::doSomething');
        $service->doSomething();
    }
}

学院君 has written 978 articles

Laravel学院院长,终身学习者

积分:112092 等级:P12 职业:手艺人 城市:杭州

2 条回复

  1. ycgambo ycgambo says:
    空对象不仅是为了避免判断一个对象是否存在,少些几行代码,更多的是给程序一个规范的对象而不是乱七八糟的东西,更好的处理默认状况或者异常。空对象应当自身处理好异常状况,而不该由其他对象做判断,这样只会复杂业务逻辑。空对象一个很好的例子就是处理业务中某些模块的启用与否。 我理解不是很深,另外我觉得如果没有深入理解一个设计思想就说它`花里胡哨。。。`,并不能让我们觉得你很强,反而觉得你很俗。
  2. z54123321 z54123321 says:
    感觉空对象确实实现了我们不需要判断这个对象到底存不存在,但是程序始终都是要往后面走的啊,我们虽然跳过了空对象处理麻烦,但是处理空对象后的操作方法比处理空对象花费的时间好像还更多,花里胡哨的模式。。。

登录后才能进行评论,立即登录?