访问者模式——元素的执行算法随着访问者改变而改变

一、基础简介

1、定义

表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。

通过这种方式,元素的执行算法可以随着访问者改变而改变。这种类型的设计模式属于行为型模式

2、使用场景

  1. 一个对象结构包含多个类型的对象,希望对这些对象实施一些依赖其具体类型的操作。在访问者中针对每一种具体的类型都提供了一个访问操作,不同类型的对象可以有不同的访问操作。
  2. 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而需要避免让这些操作“污染”这些对象的类,也不希望在增加新操作时修改这些类。访问者模式使得我们可以将相关的访问操作集中起来定义在访问者类中,对象结构可以被多个不同的访问者类所使用,将对象本身与对象的访问操作分离。
  3. 对象结构中对象对应的类很少改变,但经常需要在此对象结构上定义新的操作。

3、优缺点

优点: 1、符合单一职责原则。 2、优秀的扩展性。 3、灵活性。

缺点: 1、具体元素对访问者公布细节,违反了迪米特原则。 2、具体元素变更比较困难。 3、违反了依赖倒置原则,依赖了具体类,没有依赖抽象。

4、模式结构分析

  • Visitor(抽象访问者):抽象访问者为对象结构中每一个具体元素类ConcreteElement声明一个访问操作,从这个操作的名称或参数类型可以清楚知道需要访问的具体元素的类型,具体访问者需要实现这些操作方法,定义对这些元素的访问操作。
  • ConcreteVisitor(具体访问者):具体访问者实现了每个由抽象访问者声明的操作,每一个操作用于访问对象结构中一种类型的元素。
  • Element(抽象元素):抽象元素一般是抽象类或者接口,它定义一个accept()方法,该方法通常以一个抽象访问者作为参数。
  • ConcreteElement(具体元素):具体元素实现了accept()方法,在accept()方法中调用访问者的访问方法以便完成对一个元素的操作。
  • ObjectStructure(对象结构)对象结构是一个元素的集合,它用于存放元素对象,并且提供了遍历其内部元素的方法。它可以结合组合模式来实现,也可以是一个简单的集合对象,如一个List对象或一个Set对象。

二、实例实现

1、实例场景

比如说,男人和女人面对成功与失败,会有不同的反应或结论:

  • 1)男人成功时,背后都有一个伟大的女人;女人成功时,背后都有一群男人
  • 2)男人失败时,闷头喝酒,谁也不用劝;女人失败时,眼泪汪汪,谁也劝不了

这里,Element就是“人类”,ConcreteElementA和ConcreteElementB就是“Man”和“Woman”,而Visitor就是不同的“状态”,ConcreteVisitor就是“成功”、“失败”等具体的状态场景。

  • 使用“访问者模式”,如果需要新增一个状态(比如结婚、挫折等状态),只需要增加一个ConcreteVisitor即可。

2、Element(抽象元素)

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:06
 *
 * @description Element(抽象元素)
 */
public abstract class Person {
    // 用来获得“行为”对象的
   abstract void accept(Action visitor);
}

3、ConcreteElement(具体元素)

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:11
 * 
 * @description ConcreteElement(具体元素)
 */
public class Man extends Person {

    @Override
    void accept(Action visitor) {

        visitor.getManReact(this);
    }
}

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:11
 *
 * @description ConcreteElement(具体元素)
 */
public class Woman extends Person {

    @Override
    void accept(Action visitor) {

        visitor.getWomanReact(this);
    }
}

4、Visitor(抽象访问者)

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:07
 *
 * @description Visitor(抽象访问者)
 */
public interface Action {
    // 得到Man的反应
    void getManReact(Man concreteElementA);
    // 得到Woman的反应
    void getWomanReact(Woman concreteElementB);
}

5、双分派技术

在这里,Action的 getManReact(Man concreteElementA) 方法与 Man 里的

** void accept(Action visitor) {**

** visitor.getManReact(this);
}**

方法,充分利用双分派技术,实现了处理与数据结构的分离。

结合该实例的完整代码理解该技术:

  • 首先,在客户端,将“具体的状态”作为参数传递给“具体的Element”,完成第一分派;
// 客户端
 PersonList persons = new PersonList();
        persons.add(new Man());
        persons.add(new Woman());
        // 成功时
        Success s = new Success();
        persons.display(s);
  // persons里的方法
  // 查看显示
    public void display(Action visitor) {
        for (Person p : list) {
            p.accept(visitor);
        }
    }

  • 然后,“具体的Element”类——Man,调用作为参数的“具体的状态”中的方法“void getManReact(Man concreteElementA);” ,同时将自己(this)作为参数传递进去。完成第二次分派。
public class Man extends Person {
    @Override
    void accept(Action visitor) {
        visitor.getManReact(this);
    }
}

6、ConcreteVisitor(具体访问者)

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:14
 *
 * @description ConcreteVisitor(具体访问者)
 */
public class Failure implements Action {
    @Override
    public void getManReact(Man concreteElementA) {
        System.out.println("男人失败时,闷头喝酒,谁也不用劝");
    }

    @Override
    public void getWomanReact(Woman concreteElementB) {
        System.out.println("女人失败时,眼泪汪汪,谁也劝不了");
    }
}

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:14
 *
 * @description ConcreteVisitor(具体访问者)
 */
public class Success implements Action {
    @Override
    public void getManReact(Man concreteElementA) {
        System.out.println("男人成功时,背后都有一个伟大的女人");
    }

    @Override
    public void getWomanReact(Woman concreteElementB) {
        System.out.println("女人成功时,背后都有一群男人");
    }
}

7、ObjectStructure(对象结构)对象结构是一个元素的集合

package com.mfc.design.访问者模式;

import java.util.ArrayList;
import java.util.List;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:20
 *
 * @description ObjectStructure(对象结构):对象结构是一个元素的集合
 */
public class PersonList {
    private List<Person> list = new ArrayList<>();

    public void add(Person element) {
        list.add(element);
    }

    public void delete(Person element) {
        list.remove(element);
    }
    // 查看显示
    public void display(Action visitor) {
        for (Person p : list) {
            p.accept(visitor);
        }
    }
}

8、客户端

package com.mfc.design.访问者模式;

/**
 * @author MouFangCai
 * @date 2019/10/28 16:23
 * @description
 */
public class Client_visitor {
    public static void main(String[] args) {
        PersonList persons = new PersonList();
        persons.add(new Man());
        persons.add(new Woman());

        // 成功时
        Success s = new Success();
        persons.display(s);

        // 失败时
        Failure f = new Failure();
        persons.display(f);
    }
}

9、结果展示

男人成功时,背后都有一个伟大的女人
女人成功时,背后都有一群男人
男人失败时,闷头喝酒,谁也不用劝
女人失败时,眼泪汪汪,谁也劝不了

Process finished with exit code 0

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

推荐阅读更多精彩内容