訪問(wèn)者模式

定義:

封裝某些作用于某種數(shù)據(jù)結(jié)構(gòu)中各元素的操作泽艘,它可以在不改變數(shù)據(jù)結(jié)構(gòu)的前提下定義作用于這些元素的新的操作斥黑。

例子:
生成一個(gè)報(bào)表(員工考核)給不同領(lǐng)導(dǎo)看萝毛,員工的工作和表現(xiàn)數(shù)據(jù)已經(jīng)固定了(被訪問(wèn)者),這個(gè)時(shí)候?qū)τ诓煌I(lǐng)導(dǎo)(訪問(wèn)者)來(lái)說(shuō)言蛇,
他們考核員工的標(biāo)準(zhǔn)不同亮曹,ceo看重員工kpi,cto看重代碼和產(chǎn)品數(shù)岔留;而員工(被訪問(wèn)者)的類型又不同庵佣,每個(gè)員工都有kpi,但是程序員還有代碼量婿斥,經(jīng)理有產(chǎn)品數(shù)劝篷。

最終達(dá)到的結(jié)果是報(bào)表,數(shù)據(jù)來(lái)自被訪問(wèn)者民宿,報(bào)表就是處理訪問(wèn)者和被訪問(wèn)者的關(guān)系或者邏輯娇妓。即數(shù)據(jù)和數(shù)據(jù)邏輯分離;

訪問(wèn)者模式.png

示例代碼

class A {  
    public void method1(){  
        System.out.println("我是A");  
    }  
      
    public void method2(B b){  
        b.showA(this);  
    }  
}  
  
class B {  
    public void showA(A a){  
        a.method1();  
    }  
}  
public class Test {  
    public static void main(String[] args){  
        A a = new A();  
        a.method1();  
        a.method2(new B());  
    }  
} 

在例子中活鹰,對(duì)于類A來(lái)說(shuō)哈恰,類B就是一個(gè)訪問(wèn)者。但是這個(gè)例子并不是訪問(wèn)者模式的全部华望,雖然直觀蕊蝗,但是它的可擴(kuò)展性比較差

角色

  • 抽象訪問(wèn)者(IVisitor )
    抽象類或者接口,聲明訪問(wèn)者可以訪問(wèn)哪些元素赖舟,具體到程序中就是visit方法中的參數(shù)定義哪些對(duì)象是可以被訪問(wèn)的蓬戚。

  • 訪問(wèn)者(Visitor )
    實(shí)現(xiàn)抽象訪問(wèn)者所聲明的方法,它影響到訪問(wèn)者訪問(wèn)到一個(gè)類后該干什么宾抓,要做什么事情子漩。

  • 抽象元素類(Element )
    接口或者抽象類,聲明接受哪一類訪問(wèn)者訪問(wèn)石洗,程序上是通過(guò)accept方法中的參數(shù)來(lái)定義的幢泼。抽象元素一般有兩類方法,一部分是本身的業(yè)務(wù)邏輯讲衫,另外就是允許接收哪類訪問(wèn)者來(lái)訪問(wèn)缕棵。

  • 元素類( ConcreteElement)
    實(shí)現(xiàn)抽象元素類所聲明的accept方法,通常都是visitor.visit(this)涉兽,基本上已經(jīng)形成一種定式了招驴。

  • 結(jié)構(gòu)對(duì)象(ObjectStruture )
    一個(gè)元素的容器,一般包含一個(gè)容納多個(gè)不同類枷畏、不同接口的容器别厘,如List、Set拥诡、Map等触趴,在項(xiàng)目中一般很少抽象出這個(gè)角色氮发。

實(shí)現(xiàn)代碼

IVisitor

interface IVisitor {  
    public void visit(ConcreteElement1 el1);  
    public void visit(ConcreteElement2 el2);  
}  

Visitor

class Visitor implements IVisitor {  
  
    public void visit(ConcreteElement1 el1) {  
        el1.doSomething();  
    }  
      
    public void visit(ConcreteElement2 el2) {  
        el2.doSomething();  
    }  
} 

Element

abstract class Element {  
    public abstract void accept(IVisitor visitor);  
    public abstract void doSomething();  
}

ConcreteElement

class ConcreteElement1 extends Element {  
    public void doSomething(){  
        System.out.println("這是元素1");  
    }  
      
    public void accept(IVisitor visitor) {  
        visitor.visit(this);  
    }  
}  
  
class ConcreteElement2 extends Element {  
    public void doSomething(){  
        System.out.println("這是元素2");  
    }  
      
    public void accept(IVisitor visitor) {  
        visitor.visit(this);  
    }  
}  

ObjectStruture

class ObjectStruture {  
    public static List<Element> getList(){  
        List<Element> list = new ArrayList<Element>();  
        Random ran = new Random();  
        for(int i=0; i<10; i++){  
            int a = ran.nextInt(100);  
            if(a>50){  
                list.add(new ConcreteElement1());  
            }else{  
                list.add(new ConcreteElement2());  
            }  
        }  
        return list;  
    }  
}

Client

public class Client {  
    public static void main(String[] args){  
        List<Element> list = ObjectStruture.getList();  
        for(Element e: list){  
            e.accept(new Visitor());  
        }  
    }  
} 

優(yōu)點(diǎn)

符合單一職責(zé)原則:凡是適用訪問(wèn)者模式的場(chǎng)景中,元素類中需要封裝在訪問(wèn)者中的操作必定是與元素類本身關(guān)系不大且是易變的操作冗懦,使用訪問(wèn)者模式一方面符合單一職責(zé)原則爽冕,另一方面,因?yàn)楸环庋b的操作通常來(lái)說(shuō)都是易變的披蕉,所以當(dāng)發(fā)生變化時(shí)扇售,就可以在不改變?cè)仡惐旧淼那疤嵯拢瑢?shí)現(xiàn)對(duì)變化部分的擴(kuò)展嚣艇。
擴(kuò)展性良好:元素類可以通過(guò)接受不同的訪問(wèn)者來(lái)實(shí)現(xiàn)對(duì)不同操作的擴(kuò)展。

適用場(chǎng)景

  • 假如一個(gè)對(duì)象中存在著一些與本對(duì)象不相干(或者關(guān)系較弱)的操作华弓,為了避免這些操作污染這個(gè)對(duì)象食零,則可以使用訪問(wèn)者模式來(lái)把這些操作封裝到訪問(wèn)者中去。
  • 假如一組對(duì)象中寂屏,存在著相似的操作贰谣,為了避免出現(xiàn)大量重復(fù)的代碼,也可以將這些重復(fù)的操作封裝到訪問(wèn)者中去迁霎。

缺陷

但是吱抚,訪問(wèn)者模式并不是那么完美,它也有著致命的缺陷:增加新的元素類比較困難考廉。通過(guò)訪問(wèn)者模式的代碼可以看到秘豹,在訪問(wèn)者類中,每一個(gè)元素類都有它對(duì)應(yīng)的處理方法昌粤,也就是說(shuō)既绕,每增加一個(gè)元素類都需要修改訪問(wèn)者類(也包括訪問(wèn)者類的子類或者實(shí)現(xiàn)類),修改起來(lái)相當(dāng)麻煩涮坐。也就是說(shuō)凄贩,在元素類數(shù)目不確定的情況下,應(yīng)該慎用訪問(wèn)者模式袱讹。所以疲扎,訪問(wèn)者模式比較適用于對(duì)已有功能的重構(gòu),比如說(shuō)捷雕,一個(gè)項(xiàng)目的基本功能已經(jīng)確定下來(lái)椒丧,元素類的數(shù)據(jù)已經(jīng)基本確定下來(lái)不會(huì)變了,會(huì)變的只是這些元素內(nèi)的相關(guān)操作非区,這時(shí)候瓜挽,我們可以使用訪問(wèn)者模式對(duì)原有的代碼進(jìn)行重構(gòu)一遍,這樣一來(lái)征绸,就可以在不修改各個(gè)元素類的情況下久橙,對(duì)原有功能進(jìn)行修改俄占。

總結(jié)

正如《設(shè)計(jì)模式》的作者GoF對(duì)訪問(wèn)者模式的描述:大多數(shù)情況下,你并需要使用訪問(wèn)者模式淆衷,但是當(dāng)你一旦需要使用它時(shí)缸榄,那你就是真的需要它了。當(dāng)然這只是針對(duì)真正的大牛而言祝拯。在現(xiàn)實(shí)情況下(至少是我所處的環(huán)境當(dāng)中)甚带,很多人往往沉迷于設(shè)計(jì)模式,他們使用一種設(shè)計(jì)模式時(shí)佳头,從來(lái)不去認(rèn)真考慮所使用的模式是否適合這種場(chǎng)景鹰贵,而往往只是想展示一下自己對(duì)面向?qū)ο笤O(shè)計(jì)的駕馭能力。編程時(shí)有這種心理康嘉,往往會(huì)發(fā)生濫用設(shè)計(jì)模式的情況碉输。所以,在學(xué)習(xí)設(shè)計(jì)模式時(shí)亭珍,一定要理解模式的適用性敷钾。必須做到使用一種模式是因?yàn)榱私馑膬?yōu)點(diǎn),不使用一種模式是因?yàn)榱私馑谋锥艘蘩妫欢皇鞘褂靡环N模式是因?yàn)椴涣私馑谋锥俗杌模皇褂靡环N模式是因?yàn)椴涣私馑膬?yōu)點(diǎn)。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末众羡,一起剝皮案震驚了整個(gè)濱河市侨赡,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌纱控,老刑警劉巖辆毡,帶你破解...
    沈念sama閱讀 219,270評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異甜害,居然都是意外死亡舶掖,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,489評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門尔店,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)眨攘,“玉大人,你說(shuō)我怎么就攤上這事嚣州■晔郏” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 165,630評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵该肴,是天一觀的道長(zhǎng)情竹。 經(jīng)常有香客問(wèn)我,道長(zhǎng)匀哄,這世上最難降的妖魔是什么秦效? 我笑而不...
    開(kāi)封第一講書人閱讀 58,906評(píng)論 1 295
  • 正文 為了忘掉前任雏蛮,我火速辦了婚禮,結(jié)果婚禮上阱州,老公的妹妹穿的比我還像新娘挑秉。我一直安慰自己,他們只是感情好苔货,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,928評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布犀概。 她就那樣靜靜地躺著,像睡著了一般夜惭。 火紅的嫁衣襯著肌膚如雪姻灶。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 51,718評(píng)論 1 305
  • 那天诈茧,我揣著相機(jī)與錄音木蹬,去河邊找鬼。 笑死若皱,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的尘颓。 我是一名探鬼主播走触,決...
    沈念sama閱讀 40,442評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼疤苹!你這毒婦竟也來(lái)了互广?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,345評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤卧土,失蹤者是張志新(化名)和其女友劉穎惫皱,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體尤莺,經(jīng)...
    沈念sama閱讀 45,802評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡旅敷,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,984評(píng)論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了颤霎。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片媳谁。...
    茶點(diǎn)故事閱讀 40,117評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖友酱,靈堂內(nèi)的尸體忽然破棺而出晴音,到底是詐尸還是另有隱情,我是刑警寧澤缔杉,帶...
    沈念sama閱讀 35,810評(píng)論 5 346
  • 正文 年R本政府宣布锤躁,位于F島的核電站,受9級(jí)特大地震影響或详,放射性物質(zhì)發(fā)生泄漏系羞。R本人自食惡果不足惜郭计,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,462評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望觉啊。 院中可真熱鬧拣宏,春花似錦、人聲如沸杠人。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,011評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)嗡善。三九已至辑莫,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間罩引,已是汗流浹背各吨。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,139評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留袁铐,地道東北人揭蜒。 一個(gè)月前我還...
    沈念sama閱讀 48,377評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像剔桨,于是被迫代替她去往敵國(guó)和親屉更。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,060評(píng)論 2 355

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