設(shè)計(jì)模式-訪問者模式(二十)

  • 目的是:

封裝一些施加于某種數(shù)據(jù)結(jié)構(gòu)元素之上的操作,一旦這些操作改變,接受這個(gè)操作的數(shù)據(jù)結(jié)構(gòu)則可以保持不變

何時(shí)使用:需要對一個(gè)對象結(jié)構(gòu)中的對象進(jìn)行很多不同的并且不相關(guān)的操作,而需要避免讓這些操作"污染"這些對象的類,使用訪問者模式將這些封裝到類中

Problem:如何針對保存有不同類型對象的聚集采取某種操作?
用if幕垦?

image.png

訪問者模式適用于 數(shù)據(jù)結(jié)構(gòu)不確定的情況,它把數(shù)據(jù)結(jié)構(gòu)和操作之間的耦合解脫開傅联。

  • 使用場景: 1先改、對象結(jié)構(gòu)中對象對應(yīng)的類很少改變(也就是Node很少改變),但經(jīng)常需要在此對象結(jié)構(gòu)上定義新的操作(也就是經(jīng)常要增加Visitor)

  • 上類圖:


    訪問者角色.png
  • 優(yōu)點(diǎn):
    將有關(guān)的行為集中到一個(gè)訪問者對象中,而不是分散到一個(gè)個(gè)節(jié)點(diǎn)類中,行為集中了,相關(guān)的操作狀態(tài)(類似屬性)也積累在自己的內(nèi)部,而不是分散到很多對象中,這是有利于維護(hù)的.

  • 缺點(diǎn):
    如果要增加一個(gè)新的節(jié)點(diǎn)都要在抽象訪問者角色中增加一個(gè)新的抽象操作.

  • 代碼示例:

  1. 定義一個(gè)訪問者訪問的接口類,抽象節(jié)點(diǎn)角色
/**
 * FileName :ComputerPart
 * Author :zengzhijun
 * Date : 2018/5/28 15:34
 * Description:
 */
package com.byedbl.visitor.sample2.entity;

import com.byedbl.visitor.sample2.visitor.ComputerPartVisitor;

public interface ComputerPart {
    public void accept(ComputerPartVisitor computerPartVisitor);
}

  1. 實(shí)現(xiàn)4個(gè)具體節(jié)點(diǎn)
/**
 * FileName :Keyboard
 * Author :zengzhijun
 * Date : 2018/5/28 15:34
 * Description:
 */
package com.byedbl.visitor.sample2.entity;

import com.byedbl.visitor.sample2.visitor.ComputerPartVisitor;

public class Keyboard  implements ComputerPart {

    @Override
    public void accept(ComputerPartVisitor computerPartVisitor) {
        computerPartVisitor.visit(this);
    }
}
/**
 * FileName :Monitor
 * Author :zengzhijun
 * Date : 2018/5/28 15:35
 * Description:
 */
package com.byedbl.visitor.sample2.entity;

import com.byedbl.visitor.sample2.visitor.ComputerPartVisitor;

public class Monitor  implements ComputerPart {

    @Override
    public void accept(ComputerPartVisitor computerPartVisitor) {
        computerPartVisitor.visit(this);
    }
}

/**
 * FileName :Mouse
 * Author :zengzhijun
 * Date : 2018/5/28 15:35
 * Description:
 */
package com.byedbl.visitor.sample2.entity;

import com.byedbl.visitor.sample2.visitor.ComputerPartVisitor;

public class Mouse  implements ComputerPart {

    @Override
    public void accept(ComputerPartVisitor computerPartVisitor) {
        computerPartVisitor.visit(this);
    }
}
/**
 * FileName :Computer
 * Author :zengzhijun
 * Date : 2018/5/28 15:35
 * Description:
 */
package com.byedbl.visitor.sample2.entity;

import com.byedbl.visitor.sample2.visitor.ComputerPartVisitor;

public class Computer implements ComputerPart {

    ComputerPart[] parts;

    public Computer(){
        parts = new ComputerPart[] {new Mouse(), new Keyboard(), new Monitor()};
    }


    @Override
    public void accept(ComputerPartVisitor computerPartVisitor) {
        for (int i = 0; i < parts.length; i++) {
            parts[i].accept(computerPartVisitor);
        }
        computerPartVisitor.visit(this);
    }
}

  1. 定義訪問者抽象接口
/**
 * FileName :ComputerPartVisitor
 * Author :zengzhijun
 * Date : 2018/5/28 15:36
 * Description:
 */
package com.byedbl.visitor.sample2.visitor;

import com.byedbl.visitor.sample2.entity.Computer;
import com.byedbl.visitor.sample2.entity.Keyboard;
import com.byedbl.visitor.sample2.entity.Monitor;
import com.byedbl.visitor.sample2.entity.Mouse;

public interface ComputerPartVisitor {
    public void visit(Computer computer);
    public void visit(Mouse mouse);
    public void visit(Keyboard keyboard);
    public void visit(Monitor monitor);
}

  1. 實(shí)現(xiàn)訪問者接口
/**
 * FileName :ComputerPartDisplayVisitor
 * Author :zengzhijun
 * Date : 2018/5/28 15:36
 * Description:
 */
package com.byedbl.visitor.sample2.visitor;

import com.byedbl.visitor.sample2.entity.Computer;
import com.byedbl.visitor.sample2.entity.Keyboard;
import com.byedbl.visitor.sample2.entity.Monitor;
import com.byedbl.visitor.sample2.entity.Mouse;

public class ComputerPartDisplayVisitor implements ComputerPartVisitor {

    @Override
    public void visit(Computer computer) {
        System.out.println("Displaying Computer.");
    }

    @Override
    public void visit(Mouse mouse) {
        System.out.println("Displaying Mouse.");
    }

    @Override
    public void visit(Keyboard keyboard) {
        System.out.println("Displaying Keyboard.");
    }

    @Override
    public void visit(Monitor monitor) {
        System.out.println("Displaying Monitor.");
    }
}

  1. 客戶端用法
/**
 * FileName :VisitorPatternDemo
 * Author :zengzhijun
 * Date : 2018/5/28 15:36
 * Description:
 */
package com.byedbl.visitor.sample2;

import com.byedbl.visitor.sample2.entity.Computer;
import com.byedbl.visitor.sample2.entity.ComputerPart;
import com.byedbl.visitor.sample2.entity.Mouse;
import com.byedbl.visitor.sample2.visitor.ComputerPartDisplayVisitor;

/**
 *
 * <pre>
 * 備受爭議的訪問者模式
 * 因?yàn)樵黾右粋€(gè)Node節(jié)點(diǎn)就要在Visitor節(jié)點(diǎn)中(這里是 ComputerPartVisitor)增加一個(gè)方法支持它
 * 如果已經(jīng)實(shí)現(xiàn)了很多Visitor那不是很坑爹?當(dāng)然在java8里面可以加一個(gè)默認(rèn)的方法了,情況可能好一點(diǎn)
 * 但是還是不能掩蓋其違反了開閉原則
 *
 * 但是作為實(shí)用主義來講,只要管用,理論就可以先一邊呆著去...
 * 畢竟像中國的代碼,if else if 幾百行 也不是沒見過的
 *
 * @author : zengzhijun
 * @date : 2018/5/28 15:51
 **/
public class VisitorPatternDemo {

    public static void main(String[] args) {

        ComputerPart computer = new Computer();
        computer.accept(new ComputerPartDisplayVisitor());
        System.out.println("--------------");
        ComputerPart computerPart = new Mouse();
        computerPart.accept(new ComputerPartDisplayVisitor());
    }
}

這個(gè)例子很形象,電腦組件就這幾個(gè),但是不同的人用來干不同的事,也就是滿足對象結(jié)構(gòu)中對象對應(yīng)的類很少改變(也就是Node很少改變)蒸走,但經(jīng)常需要在此對象結(jié)構(gòu)上定義新的操作(也就是經(jīng)常要增加Visitor)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末仇奶,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子比驻,更是在濱河造成了極大的恐慌该溯,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件别惦,死亡現(xiàn)場離奇詭異狈茉,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)掸掸,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進(jìn)店門氯庆,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人扰付,你說我怎么就攤上這事点晴。” “怎么了悯周?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長陪竿。 經(jīng)常有香客問我禽翼,道長,這世上最難降的妖魔是什么族跛? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任闰挡,我火速辦了婚禮,結(jié)果婚禮上礁哄,老公的妹妹穿的比我還像新娘长酗。我一直安慰自己,他們只是感情好桐绒,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布夺脾。 她就那樣靜靜地躺著之拨,像睡著了一般。 火紅的嫁衣襯著肌膚如雪咧叭。 梳的紋絲不亂的頭發(fā)上蚀乔,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天,我揣著相機(jī)與錄音菲茬,去河邊找鬼吉挣。 笑死,一個(gè)胖子當(dāng)著我的面吹牛婉弹,可吹牛的內(nèi)容都是我干的睬魂。 我是一名探鬼主播,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼镀赌,長吁一口氣:“原來是場噩夢啊……” “哼氯哮!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起佩脊,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤蛙粘,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后威彰,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體出牧,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年歇盼,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了舔痕。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,137評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡豹缀,死狀恐怖伯复,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情邢笙,我是刑警寧澤啸如,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站氮惯,受9級特大地震影響叮雳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜妇汗,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一帘不、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧杨箭,春花似錦寞焙、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽辽狈。三九已至,卻和暖如春模她,著一層夾襖步出監(jiān)牢的瞬間稻艰,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工侈净, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留尊勿,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓畜侦,卻偏偏與公主長得像元扔,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子旋膳,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評論 2 355

推薦閱讀更多精彩內(nèi)容