概述
在这个快速教程中,我们将看一下 Null 对象模式,这是设计模式的一个特例,一个经常被我们遗忘的特例。我们将描述它的目的以及我们何时应该考虑使用它。
像往常一样,我们还将提供一个简单的代码示例。
空对象模式
在大多数面向对象的编程语言中,我们会考虑不进行 null 引用,为此我们经常被迫写空检查:
Command cmd = getCommand();
if (cmd != null) {
cmd.execute();
}
有时,如果此类 if 语句的数量变多,则代码的可读性会变得很差,难以阅读且容易出错。这是 Null 对象模式可以派上用场的时候。
Null 对象模式的目的是最小化这种空检查。相反,我们可以识别 null 行为并将其封装在客户端代码所期望的类中。这样我们就不再需要处理空引用的特殊处理。
我们可以简单地处理 null 对象,就像处理实际包含一些更复杂的业务逻辑的给定类型的任何其他实例一样,以此来保持代码的清洁。
由于 null 对象不应具有任何状态,因此无需多次创建相同的实例。因此,我们经常将 null 对象实现为单例。
Null Object Pattern 的 UML 图
让我们直观地看一下模式:
我们可以看到,会有以下参与者:
客户端需要 AbstractObject 的实例
AbstractObject 定义客户期望的执行逻辑 - 它还可以包含实现类的共享逻辑
RealObject 实现了 AbstractObject 并提供了真实的行为
NullObject 实现 AbstractObject 并提供 do nothing 行为
案例
现在我们已经清楚地了解了这个理论,让我们看一个例子。
假设一个场景,有一个消息路由器模块。每条消息都应分配有效的优先级。我们的系统应该将高优先级消息路由到 SMS 网关,而具有中等优先级的消息应该路由到 JMS 队列。
但是,有时会出现“未定义”或空优先级的消息。这些消息应该从进一步处理中丢弃。
首先,我们将创建路由器接口:
public interface Router {
void route(Message msg);
}
接下来,让我们创建上述接口的两个实现 – 负责路由到 SMS 网关的接口以及将消息路由到 JMS 队列的接口:
public class SmsRouter implements Router {
@Override
public void route(Message msg) {
// implementation details
}
}
public class JmsRouter implements Router {
@Override
public void route(Message msg) {
// implementation details
}
}
最后,让我们实现我们的 null 对象:
public class NullRouter implements Router {
@Override
public void route(Message msg) {
// do nothing
}
}
我们现在准备将所有部分组合在一起。让我们看看示例客户端代码的外观如何:
public class RoutingHandler {
public void handle(Iterable<Message> messages) {
for (Message msg : messages) {
Router router = RouterFactory.getRouterForMessage(msg);
router.route(msg);
}
}
}
我们可以看到,无论 RouterFactory 返回什么实现,我们都以相同的方式处理所有 Router 对象。这使我们能够保持代码的清洁和可读性。
何时使用空对象模式
我们应该使用 Null 对象模式,否则客户端会检查 null 只是为了跳过执行或执行默认操作。在这种情况下,我们可以将 do nothing 逻辑封装在空对象中,并将其返回给客户端而不是空值。这样,如果给定实例为 null,则客户端代码不再需要知道。
这种方法也遵循一般的面向对象原则。
为了更好地理解何时应该使用 Null 对象模式,让我们假设我们必须实现 CustomerDao 接口,定义如下:
public interface CustomerDao {
Collection<Customer> findByNameAndLastname(String name, String lastname);
Customer getById(Long id);
}
如果没有客户匹配提供的搜索条件,大多数开发人员将从 findByNameAndLastname()返回 Collections.emptyList()。这是遵循 Null 对象模式的一个很好的例子。
相反,getById() 应返回具有给定 id 的客户。调用此方法的人希望获得特定的客户实体。如果不存在这样的客户,我们应该显式返回 null 以表示提供的 ID 存在问题。
与所有其他模式一样,我们需要在盲目实现 Null 对象模式之前考虑我们的特定用例。否则,我们可能会无意中在代码中引入一些很难找到的错误。
结论
在本文中,我们了解了 Null 对象模式是什么以及何时可以使用它。我们还实现了一个设计模式的简单示例。