Android的设计模式-访问者模式

前言

Android的设计模式系列文章介绍,欢迎关注,持续更新中:

Android的设计模式-设计模式的六大原则
一句话总结23种设计模式则
创建型模式:
Android的设计模式-单例模式
Android的设计模式-建造者模式
Android的设计模式-工厂方法模式
Android的设计模式-简单工厂模式
Android的设计模式-抽象工厂模式
Android的设计模式-原型模式
行为型模式:
Android的设计模式-策略模式
Android的设计模式-状态模式
Android的设计模式-责任链模式
Android的设计模式-观察者模式
Android的设计模式-模板方法模式
Android的设计模式-迭代器模式
Android的设计模式-备忘录模式
Android的设计模式-访问者模式
Android的设计模式-中介者模式
Android的设计模式-解释器模式
Android的设计模式-命令模式
结构型模式:
Android的设计模式-代理模式
Android的设计模式-组合模式
Android的设计模式-适配器模式
Android的设计模式-装饰者模式
Android的设计模式-享元模式
Android的设计模式-外观模式
Android的设计模式-桥接模式

1.定义

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

2.介绍

  • 访问者模式属于行为型模式。
  • 访问者模式是一种将数据结构和数据操作分离的设计模式。
  • 访问者模式比较复杂,而且实际使用的地方并不多。
  • 访问者模式适用于数据结构稳定的元素操作上,一旦数据结构易变,则不适用。

3.UML类图

访问者模式UML类图.jpg
角色说明:
  • Visitor(抽象访问者):接口或者抽象类,为每一个元素(Element)声明一个访问的方法。
  • ConcreteVisitor(具体访问者):实现抽象访问者中的方法,即对每一个元素都有其具体的访问行为。
  • Element(抽象元素):接口或者抽象类,定义一个accept方法,能够接受访问者(Visitor)的访问。
  • ConcreteElementA、ConcreteElementB(具体元素):实现抽象元素中的accept方法,通常是调用访问者提供的访问该元素的方法。
  • Client(客户端类):即要使用访问者模式的地方。

4.实现

以我们平时听歌看视频为例,音乐视频网站都会提供在线播放和下载的功能,当我们有空的时候往往就选择了在线播放,比较忙的时候就选择先下载下来,有空的时候再去观看。其中,音乐视频网站就是具体的要访问的元素,闲人和忙人就是具体的访问者,闲人和忙人的访问行为是不一样的。

4.1 创建抽象元素

定义一个抽象的受访问方法以及其他公共的方法:

    public abstract class Web {
        public String name;

        public Web(String name) {
            this.name = name;
        }

        //定义一个抽象的受访问方法
        public abstract void accept(Visitor visitor);

        //下载资源
        public abstract void download();

        public String getName() {
            return name;
        }
    }
4.2 创建具体元素

实现抽象元素中的accept()方法,通常是调用访问者提供的访问该元素的方法。下面创建音乐类以及视频类,他们都有一个download()方法,但是其具体下载的内容是不一样的,同时他们也存在各自独有的方法playMusic()playVideo()

    public class Music extends Web {//音乐类,继承自Web类

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

        @Override
        public void accept(Visitor visitor) {//接受访问者的访问
            visitor.visit(this);
        }

        @Override
        public void download() {//实现父类中的公共方法
            System.out.println("下载音乐~~");
        }

        public void playMusic() {//音乐类独有方法
            System.out.println("播放音乐ing");
        }
    }

    public class Video extends Web {//视频类,继承自Web类

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

        @Override
        public void accept(Visitor visitor) {//接受访问者的访问
            visitor.visit(this);
        }

        @Override
        public void download() {//实现父类中的公共方法
            System.out.println("下载视频~~");
        }

        public void playVideo() {//视频类独有方法
            System.out.println("播放视频ing");
        }
    }
4.3 创建抽象访问者

为每一个元素声明一个访问的方法:

     public interface Visitor {
        void visit(Music music);//访问音乐类

        void visit(Video video);//访问视频类
    }
4.4 创建具体访问者

实现抽象访问者中的方法,即对每一个元素都有其具体的访问行为。下面以闲人和忙人为例:

    public class Idler implements Visitor {//闲人

        private String name;

        public Idler(String name) {
            this.name = name;
        }

        @Override
        public void visit(Music music) {
            System.out.println(name + "浏览音乐网站:" + music.getName());
            music.playMusic();
        }

        @Override
        public void visit(Video video) {
            System.out.println(name + "浏览视频网站:" + video.getName());
            video.playVideo();
        }
    }

    public class Busy implements Visitor {//忙人
        private String name;

        public Busy(String name) {
            this.name = name;
        }

        @Override
        public void visit(Music music) {
            System.out.println(name + "浏览音乐网站:" + music.getName());
            music.download();
        }

        @Override
        public void visit(Video video) {
            System.out.println(name + "浏览视频网站:" + video.getName());
            video.download();
        }
    }
4.5 创建对象结构

另外,为了方便访问多个元素,创建一个对象结构,在其内部管理元素集合,并且可以迭代这些元素供访问者访问:

     public class Websites {
        List<Web> list = new ArrayList<>();//元素集合

        public void accept(Visitor visitor) {
            Iterator<Web> iterator = list.iterator();
            while (iterator.hasNext()) {//迭代遍历访问
                iterator.next().accept(visitor);
            }
        }

        public void addWeb(Web web) {
            list.add(web);
        }
    }
4.6 客户端测试
     public void test() {
        //创建不同的元素
        Music wangyiyue = new Music("网易云音乐");
        Music kugou = new Music("酷狗");
        Video youku = new Video("优酷");
        Video iqiyi = new Video("爱奇艺");

        //放入对象结构中
        Websites websites = new Websites();
        websites.addWeb(wangyiyue);
        websites.addWeb(kugou);
        websites.addWeb(youku);
        websites.addWeb(iqiyi);

        Visitor idler1 = new Idler("闲人1号");
        websites.accept(idler1);//集合接受idler1的访问

        System.out.println("-------------------------------------");
        Visitor busy1 = new Busy("忙人2号");
        websites.accept(busy1);////集合接受busy1的访问
    }
输出结果:
 闲人1号浏览音乐网站:网易云音乐
 播放音乐ing
 闲人1号浏览音乐网站:酷狗
 播放音乐ing
 闲人1号浏览视频网站:优酷
 播放视频ing
 闲人1号浏览视频网站:爱奇艺
 播放视频ing
 -------------------------------------
 忙人2号浏览音乐网站:网易云音乐
 下载音乐~~
 忙人2号浏览音乐网站:酷狗
 下载音乐~~
 忙人2号浏览视频网站:优酷
 下载视频~~
 忙人2号浏览视频网站:爱奇艺
 下载视频~~

5. 应用场景

  • 对象结构比较稳定,很少改变,但是经常需要在此对象结构上定义新的操作行为时。
  • 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,它可以在不改变这个数据结构的前提下定义作用于这些元素的新的操作。

6. 优点

  • 各种角色各司其职,符合单一职责原则。
  • 原有的类上新增操作只需实现一个具体访问者就可以,不 必修改整个类层次,符合开闭原则。
  • 良好的扩展性,新增访问操作变得简单。
  • 数据操作和数据结构的解耦。

7. 缺点

  • 具体元素对访问者公布了实现细节,破坏了类的封装性,违反了迪米特原则。
  • 违反了依赖倒置原则,为了达到区别对待依赖了具体而不是抽象。
  • 具体元素修改的成本太大。
  • 新增具体元素困难,需要在抽象访问者角色中增加一个新的抽象操作,违反了开闭原则。

8. 其他

访问者模式实际使用中比较少,但是真正需要用到时,还是很有用的。

相关文章阅读
Android的设计模式-设计模式的六大原则
一句话总结23种设计模式则
创建型模式:
Android的设计模式-单例模式
Android的设计模式-建造者模式
Android的设计模式-工厂方法模式
Android的设计模式-简单工厂模式
Android的设计模式-抽象工厂模式
Android的设计模式-原型模式
行为型模式:
Android的设计模式-策略模式
Android的设计模式-状态模式
Android的设计模式-责任链模式
Android的设计模式-观察者模式
Android的设计模式-模板方法模式
Android的设计模式-迭代器模式
Android的设计模式-备忘录模式
Android的设计模式-访问者模式
Android的设计模式-中介者模式
Android的设计模式-解释器模式
Android的设计模式-命令模式
结构型模式:
Android的设计模式-代理模式
Android的设计模式-组合模式
Android的设计模式-适配器模式
Android的设计模式-装饰者模式
Android的设计模式-享元模式
Android的设计模式-外观模式
Android的设计模式-桥接模式

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

推荐阅读更多精彩内容