| 源碼地址 | https://github.com/DingMouRen/DesignPattern |
|---|

訪問者模式.png
- Visitor 抽象訪問者,
- ConcreteVisitor 具體訪問者,實現(xiàn)對每一個元素類訪問時所產(chǎn)生的具體行為
- Element 抽象元素,提供接受訪問者的方法,每一個元素都可以被訪問者訪問
- ElementA 具體元素,提供接受訪問方法的具體實現(xiàn)。
- ObjectStructure 對象結(jié)構(gòu) 內(nèi)部管理元素集合,可以迭代這些元素供訪問者訪問。
定義
訪問者模式封裝一些作用于某種數(shù)據(jù)結(jié)構(gòu)中的各元素的操作,可以在不改變這個數(shù)據(jù)結(jié)構(gòu)的前提下定義用于這些元素的新的操作。
使用場景
- 需要對一個對象結(jié)構(gòu)中的對象進行很多不同的且不相關(guān)的操作,同時需要避免這些操作“污染”這些對象的類,也不希望在增加新操作時修改這些類。
- 對象結(jié)構(gòu)比較穩(wěn)定,但是經(jīng)常需要在此對象結(jié)構(gòu)上定義新的操作。
舉個栗子
年終公司給員工進行績效考核,來評定績效等。評定的人是CEO和CTO,也就是訪問者,假設(shè)員工只有工程師和經(jīng)理。CEO和CTO對不同員工的關(guān)注點不同,需要對不同的員工類型進行不同的處理。
//抽象訪問者
public interface Visitor {
void visit(Staff staff);
}
//具體訪問者:ceo
public class CEOVisitor implements Visitor {
@Override
public void visit(Staff staff) {
if (staff instanceof Engineer){
Engineer engineer = (Engineer) staff;
System.out.println("CEO對"+engineer.name+"數(shù)據(jù)訪問并作出自己的評語");
}else if (staff instanceof Manager){
Manager manager = (Manager) staff;
System.out.println("CEO對"+manager.name+"數(shù)據(jù)訪問并作出自己的評語");
}
}
//抽象元素類:員工基類
public abstract class Staff {
public String name;
public int kpi;//員工kpi
public Staff(String name) {
this.name = name;
kpi = new Random().nextInt(10);
}
//接收visitor的訪問
public abstract void accept(Visitor visitor);
}
//具體元素類:工程師
public class Engineer extends Staff {
public Engineer(String name) {
super(name);
}
@Override
public void accept(Visitor visitor) {
visitor.visit(this);
}
//工程師一年寫的代碼量
public int getCodeLines(){
return new Random().nextInt(10 * 1000);
}
}
//對象結(jié)構(gòu)類
public class BusinessReport {
List<Staff> staffs = new LinkedList<>();
public BusinessReport() {
staffs.add(new Manager("王經(jīng)理"));
staffs.add(new Manager("向經(jīng)理"));
staffs.add(new Engineer("工程師--Jim"));
staffs.add(new Engineer("工程師--Jack"));
staffs.add(new Engineer("工程師--Jvm"));
}
//為訪問者展示報表
public void showReport(Visitor visitor){
for (Staff staff : staffs){
staff.accept(visitor);
}
}
}
使用
public static void main(String[] args) {
//構(gòu)建報表對象
BusinessReport report = new BusinessReport();
//設(shè)置訪問者
report.showReport(new CEOVisitor());
System.out.println("- - - - ");
report.showReport(new CTOVisitor());
}
總結(jié)
訪問者模式具有優(yōu)秀的擴展性,使得數(shù)據(jù)結(jié)構(gòu)和作用于結(jié)構(gòu)上的操作解耦,操作集合可以獨立變化,但是具體元素變更時導(dǎo)致修改成本大。