一、介绍
访问者模式是一种将数据操作与数据结构分离的设计模式
基本想法:
(1)软件系统中拥有一个由许多对象构成的、比较稳定的对象结构,这些对象的类都拥有一个accept方法用来接受访问者对象的访问。访问者是一个接口,它拥有一个visit方法,这个方法对访问到的对象结构中不同类型的元素做出不同的处理。
(2)在对象结构的一次访问过程中,我们遍历整个对象结构,对每一个元素都实施accept方法,在每一个元素的acept方法中调用访问者的visit方法,从而使访问者得以处理对象结构的每一个元素,我们可以针对对象结构设计不同的访问者类来完成不同的操作,达到区别对待的效果。
二、定义及使用场景
定义及使用场景
定义:封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个数据结构的前提下定义作用于这些元素的新的操作。
可以对定义这么理解:有这么一个操作,它是作用于一些元素之上的,而这些元素属于某一个对象结构。同时这个操作是在不改变各元素类的前提下,在这个前提下定义新操作是访问者模式精髓中的精髓。
使用场景:
(1)对象结构比较稳定,但经常需要在此对象结构上定义新的操作。
(2)需要对一个对象结构中的对象进行很多不同的且不相关的操作,而需要避免这些操作“污染”这些对象的类,也不希望在增加新操作时修改这些类。
三、类图
图解:
抽象访问者(Visitor)角色:为该对象结构中具体元素角色声明一个访问操作接口,用来代表为对象结构添加的新功能,理论上可以代表任意的功能。
具体访问者(ConcreteVisitor)角色:实现每个由抽象访问者角色(Visitor)声明的操作接口。
抽象元素(Element)角色:定义一个accept()操作,它以一个访问者(Visitor)对象作为参数。
具体元素(ConcreteElement)角色:实现由抽象元素(Element)角色提供的accept()操作。
对象结构(ObjectStructure)角色:这是使用访问者模式必备的角色。一般具备以下特征:
(1)能枚举它的元素。
(2)可以提供一个高层的接口以允许该访问者访问它的元素。
(3)可以是一个复合(组合模式)或是一个集合,如一个列表或一个无序集合
1.首先定义一个接口来代表要新加入的功能,称为访问者,访问对象结构中的对象。
//访问者接口
interface Visitor{
// 访问concreteElementA,相当于为concreteElementA添加新功能
public void visitConcreteElementA(ConcreteElementA elementA);
// 访问concreteElementB,相当于为concreteElementB添加新功能
public void visitConcreteElementB(ConcreteElementB elementB);
}
2.访问者具体实现
//具体访问者A
class concreteVisitorA implements Visitor{
public void visitConcreteElementA(ConcreteElementA elementA){
// 把访问ConcreteElementA时,需要执行的功能在这里实现,可能需要访问元素已有的功能,比如operationA()
System.out.println("concreteVisitorA访问====》ConcreteElementA对象");
}
public void visitConcreteElementB(ConcreteElementB elementB){
// 把访问ConcreteElementB时,需要执行的功能在这里实现,可能需要访问元素已有的功能,比如operationB()
System.out.println("concreteVisitorA访问====》ConcreteElementB对象");
}
}
//具体访问者B
class concreteVisitorB implements Visitor{
public void visitConcreteElementA(ConcreteElementA elementA){
// 把访问ConcreteElementA时,需要执行的功能在这里实现,可能需要访问元素已有的功能,比如operationA()
System.out.println("concreteVisitorB访问====》ConcreteElementA对象");
}
public void visitConcreteElementB(ConcreteElementB elementB){
// 把访问ConcreteElementB时,需要执行的功能在这里实现,可能需要访问元素已有的功能,比如operationB()
System.out.println("concreteVisitorB访问====》ConcreteElementB对象");
}
}
3.抽象元素Element
//抽象元素Element的定义
abstract class Element{
// 接受访问者的访问
public abstract void accept(Visitor visitor);
}
4.具体的Element
//具体ElementA
class ConcreteElementA extends Element{
public void accept(Visitor visitor){
// 回调访问者对象的相应方法
visitor.visitConcreteElementA(this);
}
// 表示元素已有的功能实现
public void operationA(){
System.out.println("执行ConcreteElementA已有的operationA方法");
}
}
//具体ElementB
class ConcreteElementB extends Element{
public void accept(Visitor visitor){
// 回调访问者对象的相应方法
visitor.visitConcreteElementB(this);
}
// 表示元素已有的功能实现
public void operationB(){
System.out.println("执行ConcreteElementB已有的operationB方法");
}
}
5.对象结构
//对象结构,通常在这里对元素对象进行遍历,让访问者能够访问到所有的元素
class ObjectStructure{
// 表示对象结构
private List<Element> list =new ArrayList<Element>();
// 组建对象结构,向对象结构中添加元素
public void addElement(Element element){
this.list.add(element);
}
// 提供给客户端操作的方法,让访问者对对象结构中的所有元素进行访问
public void handleRequest(Visitor visitor){
// 循环对象结构中的元素,进行访问
for (Element element:list){
element.accept(visitor);
}
}
}
6.客户类
public class Client {
public static void main(String[] args) {
// 创建对象结构
ObjectStructure os=new ObjectStructure();
// 为对象结构中添加元素对象
os.addElement(new ConcreteElementA());
os.addElement(new ConcreteElementB());
// 创建访问者
Visitor visitor =new concreteVisitorA();
// 调用对象结构的业务处理方法
os.handleRequest(visitor);
}
}
四.应用
一个商场(SuperMarket),通常都会包括:商店(Store)、监控室(MonitoringRoom)、卫生间(WaterCloset)。商场的访问者大致可以分为两大类:顾客(Customer)、商场工作人员(MarketStaff)。顾客可以逛商店、上卫生间,但却不能进入监控室;工作人员可以进入监控室、上卫生间,但却不能像顾客一样逛商店,也就是说对于商场的同一个地点,不同的访问者有不同的行为权限,而且访问者的种类很有可能需要根据时间的推移发生变化(没准哪天,工商局的人要来视察呢!此时就需要增加工商局人员的访问者了)。
1.类图
2.实现
package 访问者模式案例2;
import java.util.ArrayList;
import java.util.List;
//访问者接口
interface Visitor{
public void visitWaterCloset(WaterCloset waterCloset);
public void visitMonitoringRoom(MonitoringRoom monitoringRoom);
public void visitStore(Store store);
}
//实现类
class MarketStaff implements Visitor{
public void visitWaterCloset(WaterCloset waterCloset){
System.out.println("工作人员进入厕所");
waterCloset.wash();
}
public void visitMonitoringRoom(MonitoringRoom monitoringRoom){
System.out.println("工作人员进入监控室");
monitoringRoom.monitor();
}
public void visitStore(Store store){
System.out.println("工作人员不能逛商店哦");
}
}
class Customer implements Visitor{
public void visitWaterCloset(WaterCloset waterCloset){
System.out.println("顾客进入厕所");
waterCloset.wash();
}
public void visitMonitoringRoom(MonitoringRoom monitoringRoom){
System.out.println("顾客不能查看监控哦");
}
public void visitStore(Store store){
System.out.println("顾客进入商店");
store.shopping();
}
}
//抽象元素
abstract class Place{
public abstract void accept(Visitor visitor);
}
//具体元素
class WaterCloset extends Place{
public void accept(Visitor visitor){
visitor.visitWaterCloset(this);
}
// 自己的方法
public void wash(){
System.out.println("洗手手");
}
}
class Store extends Place{
public void accept(Visitor visitor){
visitor.visitStore(this);
}
// 自己的方法
public void shopping(){
System.out.println("购物");
}
}
class MonitoringRoom extends Place{
public void accept(Visitor visitor){
visitor.visitMonitoringRoom(this);
}
// 自己的方法
public void monitor(){
System.out.println("查看监控");
}
}
//对象结构类
class SuperMarket{
// 对象结构
private List<Place> list =new ArrayList<>();
// 对象结构组建方法
public void add(Place place){
this.list.add(place);
}
// 为客户端提供一个遍历所有对象结构元素的方法,输入访问者参数
public void handleRequest(Visitor visitor){
for (Place place:list){
place.accept(visitor);
}
}
}
public class Client {
public static void main(String[] args) {
// 创建商场
SuperMarket superMarket=new SuperMarket();
// 添加元素
superMarket.add(new WaterCloset());
superMarket.add(new MonitoringRoom());
superMarket.add(new Store());
// 提供访问者
Visitor visitor1=new Customer();
Visitor visitor2=new MarketStaff();
superMarket.handleRequest(visitor1);
superMarket.handleRequest(visitor2);
}
}