访问者模式
访问者模式是一种将数据操作与数据结构分离的设计模式,他是设计模式中23种设计模式中最复杂的一个,但它的使用频率并不高。《设计模式》的作者GOF对访问者模式的描述:大多数情况下,你不需要使用访问者模式,但是当你一旦需要使用它时,那你就是真的需要它了。
定义
封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个数据结构的前提下定义作用于这个元素的新的操作。
使用场景
1、对象结构比较稳定,但经常需要在此对象结构上定义新的操作。
2、需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而需要避免这些操作“污染”这些对象的类,也不希望再增加新操作是修改这些类。
UML图:
代码:
Staff.java
public abstract class Staff {
public String name;
public int kpi;
public Staff(String aName){
this.name=aName;
kpi=new Random().nextInt(10);
}
public abstract void accept(Visitor visitor);
}
Engineer.java
public class Engineer extends Staff{
public Engineer(String aName){
super(aName);
}
@Override
public void accept(Visitor visitor){
visitor.visit(this);
}
public int getCodeLines(){
return new Random().nextInt(10*1000);
}
}
BusinessReport.java
public class BusinessReport {
List<Staff> mStaffs=new LinkedList<Staff>();
public BusinessReport(){
mStaffs.add(new Manager("王经理"));
mStaffs.add(new Engineer("工程师-Shawn.Xiong"));
mStaffs.add(new Engineer("工程师-Kael"));
mStaffs.add(new Engineer("工程师-Chaossss"));
mStaffs.add(new Engineer("工程师-Tiiime"));
}
public void showReport(Visitor visitor){
for(Staff staff: mStaffs){
staff.accept(visitor);
}
}
}
Visitor.java
public interface Visitor {
public void visit(Engineer engineer);
public void visit(Manager mgr);
}
CEOVisitor.java
public class CEOVisitor implements Visitor {
@Override
public void visit(Engineer engineer){
System.out.println("工程师 : "+engineer.name+", KPI : "+ engineer.kpi);
}
@Override
public void visit(Manager manager){
System.out.println("经理 : "+manager.name+", KPI : "+ manager.kpi+", 新产品数量 : "
+ manager.getProducts());
}
}
CTOVisitor.java
public class CTOVisitor implements Visitor {
@Override
public void visit(Engineer engineer){
System.out.println("工程师 : "+engineer.name+", 代码行数 : "+ engineer.getCodeLines());
}
@Override
public void visit(Manager manager){
System.out.println("经理 : "+manager.name+", KPI : "+ manager.getProducts());
}
}
Manager.java
public class Manager extends Staff {
private int products;
public Manager(String aName){
super(aName);
products=new Random().nextInt(10);
}
@Override
public void accept(Visitor visitor){
visitor.visit(this);
}
public int getProducts(){
return products;
}
}
Client.java
public class Client {
public static void main(String[] args){
BusinessReport report=new BusinessReport();
System.out.println("========给CEO看的报表=========");
report.showReport(new CEOVisitor());
System.out.println("========给CTO看的报表========");
report.showReport(new CTOVisitor());
}
}
运行结果: