访问者模式

定义

封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个暑假结构的前提下定义作用于这些元素的新的操作

使用场景

  • 对象结构比较稳定,但经常需要再此对象上定义新的操作
  • 需要对一个对象结构中的对象进行很多不同的并且不想关的操作,而需要避免这些操作“污染”这些对象的类,也不希望在新增操作时修改这些类

结构

  • visitor 接口或抽象类
    定义了对每一个元素访问的行为,它的参数就是可以访问的元素

  • ConcreteVisitor 具体访问者
    给出每一个元素类访问时产生的具体行为

  • Element 元素接口或抽象接口
    定义了一个接受访问者的方法,其意义是指每个元素都要可以被访问者访问

  • ElementA、ElementB 具体元素类
    提供接受访问者的具体实现

  • ObjectStructure 定义当中所提到的对象结果
    它内部管理了元素集合,并且可以迭代这些元素供访问者访问

简单实现


public interface Visitor {

    void visit(Engineer engineer);

    void visit(Manager manager);
}

Visitor定义了一个visitor接口,有2个visit函数,对于Engineer、Manager的访问会调用2个不同的方法,以此达到差异化处理

public abstract class Staff {

    public String name;

    public int kpi;

    public Staff(String name) {
        this.name = name;
        kpi = new Random().nextInt(10);
    }

    public abstract void accept(Visitor visitor);
}

Staff为员工基类,定义了一个accept方法用来表示接受访问者访问,由子类具体实现

public class Manager extends Staff {
    private int products;

    public Manager(String name) {
        super(name);
        products = new Random().nextInt(10);
    }

    @Override
    public void accept(Visitor visitor) {
        visitor.visit(this);
    }

    public int getProducts() {
        return products;
    }
}

Manager为经理类

public class Engineer extends Staff {


    public Engineer(String name) {
        super(name);
    }

    @Override
    public void accept(Visitor visitor) {
        visitor.visit(this);
    }


    /**
     * @return 工程师这一年的代码数量
     */
    public int getCodeLines() {
        return new Random().nextInt(10 * 10000);
    }

}

Engineer为工程师类

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 + " ,新产品数量 :" + manager.getProducts());
    }
}

CTOVisitor为CTO访问者,关心技术贡献

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());
    }
}

CEOVisitor为CEO访问者 只关注业绩

public class BusinessReport {

    List<Staff> mStaff = new LinkedList<>();

    public BusinessReport() {
        mStaff.add(new Manager("王经理"));
        mStaff.add(new Engineer("工程师-Shawn.Xiong"));
        mStaff.add(new Engineer("工程师-Kael"));
        mStaff.add(new Engineer("工程师-Jack"));
        mStaff.add(new Engineer("工程师-Rose"));
    }

    /**
     * 为访问者展示报表
     * @param visitor
     */
    public void showReport(Visitor visitor) {
        for (Staff staff : mStaff) {
            staff.accept(visitor);
        }
    }
}

BusinessReport为员工业务报表,访问者可以通过showReport函数查看所有员工的业绩

public class Client {
    public static void main(String[] args) {
        BusinessReport report = new BusinessReport();

        System.out.println("---------- 给 CEO 看的报表 -------------");

        report.showReport(new CEOVisitor());

        System.out.println("\n---------- 给 CTO 看的报表 -------------");

        report.showReport(new CTOVisitor());
    }
}

运行结果:

---------- 给 CEO 看的报表 -------------
经理 :王经理 , KPI0 ,新产品数量 :3
工程师 :工程师-Shawn.Xiong , KPI6
工程师 :工程师-Kael , KPI4
工程师 :工程师-Jack , KPI1
工程师 :工程师-Rose , KPI2

---------- 给 CTO 看的报表 -------------
经理 :王经理 ,新产品数量 :3
工程师 :工程师-Shawn.Xiong , 代码行数67720
工程师 :工程师-Kael , 代码行数37258
工程师 :工程师-Jack , 代码行数96044
工程师 :工程师-Rose , 代码行数3708
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容