訪問者模式(為一個對象的組合增加新的能力)

源碼地址 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)致修改成本大。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容