十一个行为模式之访问者模式(Visitor Pattern)
定义:
提供一个作用于某对象结构(通常是一个对象集合)的操作的接口,使得在添加新的操作或者在添加新的元素时,不需要修改原有系统,就可以对各个对象进行操作。
结构图:
- Visitor:抽象访问者类,对元素的每一种类型声明一个访问操作,从这个操作的名称或者参数类型可以清楚的知道需要被访问的元素的类型。
- ConcreteVisitor:具体访问者类型,实现对每一个元素的访问的具体操作。
- Element:抽象元素类,所有被访问元素的抽象类或者接口,定义了一个accept方法,该方法以访问者为参数,在方法中调用访问者定义的visit方法。
- ConcreteElement:具体元素类,调用accept方法,指定visit具体的对应自己的操作方法。
- ObjectStructure:对象结构类,一个元素的集合,可以采用组合模式对Element进行组织管理。当该结构调用accept()时,将遍历或递归所有集合内元素调用accept()方法。
双重分派
一个方法根据两个宗量的类型来决定执行不同的代码,这就是“双重分派”。Java语言不支持动态的多分派,也就意味着Java不支持动态的双分派。但是通过使用设计模式,也可以在Java语言里实现动态的双重分派。
在访问者这模式中,分别通过访问者的visit方法和元素的accept方法来实现双重分派,使得面向抽象编程时也可以动态的根据元素类型和访问者类型调用合适的方法。
访问者代码:
abstract class Visitor
{
public abstract void visit(ConcreteElementA elementA);
public abstract void visit(ConcreteElementB elementB);
public void visit(ConcreteElementC elementC)
{
//元素ConcreteElementC操作代码
}
}
class ConcreteVisitor extends Visitor
{
public void visit(ConcreteElementA elementA)
{
//元素ConcreteElementA操作代码
}
public void visit(ConcreteElementB elementB)
{
//元素ConcreteElementB操作代码
}
}
元素类代码:
interface Element
{
public void accept(Visitor visitor);
}
class ConcreteElementA implements Element
{
public void accept(Visitor visitor)
{
visitor.visit(this);
}
public void operationA()
{
//业务方法
}
}
具体双重分派的解释:
《JAVA与模式》之访问者模式
与组合模式联合使用
在设计对象结构类的时候可以引入组合模式来完成复杂对象的设计。
优点:
- 增加新的访问者十分容易,无需更改源码,就可以更改对象的处理方式,符合“开闭原则”。
- 将有关元素处理的方法都集中在访问者类当中,有利于访问者的复用,或者相同的对象结构有不同的访问者。
- 客户端只需要针对抽象编程,不需要关心具体元素分类。