访问者模式
Visitor模式也叫访问者模式,是行为模式之一,它分离对象的数据和行为,使用Visitor模式,可以不修改已有类的情况下,增加新的操作。
比如有一个公园,有一到多个不同的组成部分;该公园存在多个访问者:清洁工A负责打扫公园的A部分,清洁工B负责打扫公园的B部分,公园的管理者负责检点各项事务是否完成,上级领导可以视察公园等等。也就是说,对于同一个公园,不同的访问者有不同的行为操作,而且访问者的种类也可能需要根据时间的推移而变化(行为的扩展性)。根据软件设计的开闭原则(对修改关闭,对扩展开放),我们利用访问者模式进行修改。
一、结构与角色
- 访问者角色(Visitor):为该对象结构中具体元素角色声明一个访问操作接口。该操作接口的名字和参数标识了发送访问请求给具体访问者的具体元素角色。这样访问者就可以通过该元素角色的特定接口直接访问它。
- 具体访问者角色(Concrete Visitor):实现每个由访问者角色(Visitor)声明的操作,例如应用场景的清洁工A,清洁工B。
- 元素角色(Element):定义一个Accept操作,它以一个访问者为参数,例如应用场景的ParkElement,是公园每一部分的抽象。
- 具体元素角色(Concrete Element):实现由元素角色提供的Accept操作,例如应用场景里面的A部分,B部分。
- 对象结构角色(Object Structure):这是使用访问者模式必备的角色。它要具备以下特征:能枚举它的元素;可以提供一个高层的接口以允许该访问者访问它的元素;可以是一个复合(组合模式)或是一个集合,如一个列表或一个无序集合,例如应用场景里面的Park。
二、示例代码
2.1 ParkElement
//公园每部分的抽象
public interface ParkElement{
//用来接纳访问者
public void accept(Visitor visitor);
}
2.2 ParkA
public class ParkA implements ParkElement {
public void accept(Visitor visitor) {
visitor.visit(this);
}
}
2.3 ParkB
public class ParkB implements ParkElement {
public void accept(Visitor visitor) {
visitor.visit(this);
}
}
2.4 Park
public class Park implements ParkElement {
private ParkA parkA;
private ParkB parkB;
public Park(){
this.parkA = new ParkA();
this.parkB = new ParkB();
}
public void accept(Visitor visitor) {
visitor.visit(this);
parkA.accept(visitor);
parkB.accept(visitor);
}
}
2.5 Visitor
//访问者
public interface Visitor {
public void visit(Park park);
public void visit(ParkA parkA);
public void visit(ParkB parkB);
}
2.6 CleanerA
//清洁工A,负责ParkA的卫生情况
public class CleanerA implements Visitor {
public void visit(Park park) {
}
public void visit(ParkA parkA) {
System.out.println("清洁工A:完成公园A部分的卫生");
}
public void visit(ParkB parkB) {
}
}
2.7 CleanerB
//清洁工B,负责ParkB的卫生情况
public class CleanerB implements Visitor {
public void visit(Park park) {
}
public void visit(ParkA parkA) {
}
public void visit(ParkB parkB) {
System.out.println("清洁工B:完成公园B部分的卫生");
}
}
2.8 CleanerManager
public class CleanerManager implements Visitor {
public void visit(Park park) {
System.out.println("管理员:负责公园卫生检查");
}
public void visit(ParkA parkA) {
System.out.println("管理员:负责公园A卫生检查");
}
public void visit(ParkB parkB) {
System.out.println("管理员:负责公园B卫生检查");
}
}
2.9 MainClass
public class MainClass {
public static void main(String[] args){
Park park = new Park();
CleanerA cleanerA = new CleanerA();
CleanerB cleanerB = new CleanerB();
CleanerManager cleanerManager = new CleanerManager();
park.accept(cleanerA);
park.accept(cleanerB);
park.accept(cleanerManager);
}
}
//运行结果
清洁工A:完成公园A部分的卫生
清洁工B:完成公园B部分的卫生
管理员:负责公园卫生检查
管理员:负责公园A卫生检查
管理员:负责公园B卫生检查