青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

積木

No sub title

  C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
  140 Posts :: 1 Stories :: 11 Comments :: 0 Trackbacks

常用鏈接

留言簿(1)

我參與的團隊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

轉載自:http://patmusing.blog.163.com/blog/static/1358349602010150249596/

表示一個作用于某對象結構中的各元素的操作。它可以在不改變各元素的類的前提下定義作用于這些元素的新的操作。

“Represent an operation to be performed on the elements of an object structure. Visitor lets you define a new operation without changing the classes of the elements on which it operates.” – GoF

類層次結構中可能經常由于引入新的操作,從而將類型變得很脆弱。比如假定三個類Rectangle、TriangleLine繼承自抽象類Shape,現在由于某種原因需要在抽象類Shape中增加一個abstract (或者pure virtual)方法,如果使用通常的方法,那么它所有的子類,即Rectangle、TriangleLine都需要修改。

26. C++實現Behavioral - Visitor模式 - 玄機逸士 - 玄機逸士博客

由于需求的改變,某些類層次結構中常常需要增加新的行為(方法),如果直接在基類中做這樣的更改,將會給子類帶來很繁重的變更負擔,甚至破壞原有的設計。Visitor設計模式就是在不更改類層次結構的前提下,在運行時根據需要動態地為類層次結構上的各個類動態添加新的操作(方法)。

下面是Visitor模式的UML類圖:

26. C++實現Behavioral - Visitor模式 - 玄機逸士 - 玄機逸士博客

Visitor設計模式要解決的問題:

- 類層次結構不變,但類的方法需要增加;

- 預料到基類有更改,但不知道是何種具體更改。

要增加一個方法,實際上就是增加一個Visitor的子類。

業務實例:

RectangleCircle兩種形狀,在不改動他們代碼的情況下,分別為它們增加填充(Fill)和附加文字(AddText)的功能。

實現上面業務實例的C++代碼:

// Visitor.h

#include <iostream>

using namespace std;

class Visitor;

class Shape

{

public:

// draw為已經確知需要的方法

virtual void draw() = 0;

// 1. 預料將來可能會引入新的方法,因此在這里預留一個accept方法

// 2. 可以通過下面的accept方法,為Shape的子類增加一個或多個方法

virtual void accept(Visitor *visitor) = 0;

public:

virtual ~Shape()

{

cout << "in the destructor of Shape..." << endl;

}

};

class Rectangle : public Shape

{

public:

void draw();

void accept(Visitor *visitor);

public:

~Rectangle()

{

cout << "in the destructor of Rectangle..." << endl;

}

};

class Circle : public Shape

{

public:

void draw();

void accept(Visitor *visitor);

public:

~Circle()

{

cout << "in the destructor of Circle..." << endl;

}

};

class Visitor

{

public:

// 注意此處參數為具體類,而非抽象類Shape

// Shape有多少個子類,就需要在此處重載多少次visit方法。

// 除非使用RTTI,則可以只有一個visit方法,并且該方法的參數類型為抽象的Shape*,同時需要在visit方法中確定,

// 傳入的參數到底是Shape的哪個具體的子類。

virtual void visit(Rectangle *shape) = 0;

virtual void visit(Circle *shape) = 0;

public:

virtual ~Visitor()

{

cout << "in the destructor of Visitor..." << endl;

}

};

// 假定要為Shape的子類增加一個填充(Fill)操作,那么就為Visitor增加一個子類,即FillVisitor

class FillVisitor : public Visitor

{

public:

void visit(Rectangle *shape)

{

// 為類Rectangle增加一個方法,由于實現這個新方法可能需要用到原來對象中的屬性或者方法,因此需要

// 將自身(this)傳到這里。理論上而言,如果不用到原來對象的原有的屬性或者方法,那么visit方法是可以

// 沒有參數的;另外一種情況就是,如果必要,visit方法也可以擁有多個參數

cout << "Here is the newly added Fill() method for class Rectangle." << endl;

}

void visit(Circle *shape)

{

// 為類Circle增加一個方法,由于實現這個新方法可能需要用到原來對象中的屬性或者方法,因此需要

// 將自身(this)傳到這里。理論上而言,如果不用到原來對象的原有的屬性或者方法,那么visit方法是可以

// 沒有參數的;另外一種情況就是,如果必要,visit方法也可以擁有多個參數

cout << "Here is the newly added Fill() method for class Circle." << endl;

}

public:

~FillVisitor()

{

cout << "in the destructor of FillVisitor..." << endl;

}

};

// 假定要為Shape的子類增加一個附加文字(AddText)操作,那么就為Visitor增加一個子類,即AddTextVisitor

class AddTextVisitor : public Visitor

{

public:

void visit(Rectangle *shape)

{

// 為類Rectangle增加一個方法,由于實現這個新方法可能需要用到原來對象中的屬性或者方法,因此需要

// 將自身(this)傳到這里。理論上而言,如果不用到原來對象的原有的屬性或者方法,那么visit方法是可以

// 沒有參數的;另外一種情況就是,如果必要,visit方法也可以擁有多個參數

cout << "Here is the newly added AddText() method for class Rectangle." << endl;

}

void visit(Circle *shape)

{

// 為類Circle增加一個方法,由于實現這個新方法可能需要用到原來對象中的屬性或者方法,因此需要

// 將自身(this)傳到這里。理論上而言,如果不用到原來對象的原有的屬性或者方法,那么visit方法是可以

// 沒有參數的;另外一種情況就是,如果必要,visit方法也可以擁有多個參數

cout << "Here is the newly added AddText() method for class Circle." << endl;

}

public:

~AddTextVisitor()

{

cout << "in the destructor of AddTextVisitor..." << endl;

}

};

// Visitor.cpp

#include "Visitor.h"

void Rectangle::draw()

{

cout << "Implement method Rectangle::draw() here!" << endl;

}

void Rectangle::accept(Visitor *visitor)

{

visitor->visit(this); // 轉發到visitorvisit方法,并將this作為參數傳遞過去

}

void Circle::draw()

{

cout << "Implement method Circle::draw() here!" << endl;

}

void Circle::accept(Visitor *visitor)

{

visitor->visit(this); // 轉發到visitorvisit方法,并將this作為參數傳遞過去

}

// PatternClient.cpp

#include "Visitor.h"

int main(int argc, char **argv)

{

Visitor *visitor1 = new FillVisitor();

Shape *rectangle = new Rectangle();

// 調用該形狀已經實現的方法

rectangle->draw();

// 通過visitor1調用新增加的方法

rectangle->accept(visitor1);

Shape *circle = new Circle();

// 調用該形狀已經實現的方法

rectangle->draw();

// 通過visitor1調用新增加的方法

circle->accept(visitor1);

cout << "~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" << endl;

Visitor *visitor2 = new AddTextVisitor();

// 通過visitor2調用新增加的方法

rectangle->accept(visitor2);

// 通過visitor2調用新增加的方法

circle->accept(visitor2);

cout << "~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" << endl;

delete visitor1;

delete visitor2;

delete rectangle;

delete circle;

return 0;

}

運行結果:

Implement method Rectangle::draw() here!

Here is the newly added Fill() method for class Rectangle.

Implement method Rectangle::draw() here!

Here is the newly added Fill() method for class Circle.

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Here is the newly added AddText() method for class Rectangle.

Here is the newly added AddText() method for class Circle.

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

in the destructor of FillVisitor...

in the destructor of Visitor...

in the destructor of AddTextVisitor...

in the destructor of Visitor...

in the destructor of Rectangle...

in the destructor of Shape...

in the destructor of Circle...

in the destructor of Shape...

結果符合預期。

由上面的代碼可以很清楚地知道,我們如果要為RectangleCircle增加新方法,只需要多增加一個Visitor的子類即可,而RectangleCircle這兩個類本身的代碼,沒無任何改動。

上面代碼對應的UML類圖:

26. C++實現Behavioral - Visitor模式 - 玄機逸士 - 玄機逸士博客

雙向分發(Double Dispatch)

- accept(Visitor *visitor)方法在Shape的子類 (Rectangle)中;

- visit(Rectangle *rectangle)visit方法在Visitor的子類 (FillVisitor)中;

- 兩處重要的多態特性:

n 調用accept方法的對象之多態辨析,即是誰調用accept方法?(在本例中,是Rectangle還是Circle)

n accept方法的參數之多態辨析,即accept方法的參數到底是那個Visitor的子類

正是這兩處多態特性的使用,就形成了所謂的雙向分發(Double Dispatch)機制,即首先將visitor作為參數傳遞給Shape的子類(在本例中即RectangleCircle);然后在Shape子類的accept方法中調用visitorvisit方法,而且visit方法的參數是this,這樣Shape子類就將自己傳遞到了visitor類的visit方法中,然后在visitorvisit方法中為Shape的子類增加需要的新的行為。這就是雙向分發的核心思想,即visitor被傳遞(分發)了一次,即visitor被傳遞到Shape子類中一次,Shape的子類也被傳遞(分發)了一次,即Shape的子類被傳遞到visitor中一次。

從更高層次來看這個問題:Shape中的accept()Visitor作為參數,Visitor中的visitShape作為參數。

上面給出代碼的調用順序:

26. C++實現Behavioral - Visitor模式 - 玄機逸士 - 玄機逸士博客

posted on 2013-03-08 16:01 Jacc.Kim 閱讀(263) 評論(0)  編輯 收藏 引用 所屬分類: 設計模式
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品免费观看视频| 日韩视频免费看| 亚洲巨乳在线| 午夜精品久久久久影视| aaa亚洲精品一二三区| 欧美一区二区三区四区在线| 亚洲精品欧美精品| 美女精品网站| 毛片基地黄久久久久久天堂| 国产精品美女| 一本久道久久综合中文字幕| 最新日韩欧美| 老司机精品视频网站| 久久久91精品| 国产亚洲欧美日韩日本| 亚洲午夜久久久久久久久电影院 | 欧美电影专区| 久久久久欧美| 国产亚洲人成网站在线观看| 一区二区欧美亚洲| 国产精品99久久久久久久久久久久| 美国十次成人| 在线观看av不卡| 久久国产精品久久久久久| 亚洲专区欧美专区| 欧美日韩和欧美的一区二区| 亚洲伦伦在线| 亚洲一区二区精品视频| 欧美视频官网| 亚洲先锋成人| 久久久999| 亚洲风情在线资源站| 麻豆成人av| 91久久久久久久久久久久久| 亚洲日韩欧美视频一区| 欧美久久久久免费| 一本色道久久综合狠狠躁篇怎么玩 | 毛片基地黄久久久久久天堂| 久久综合电影一区| 亚洲国产精品成人久久综合一区| 亚洲国产一二三| 9l国产精品久久久久麻豆| 欧美日韩国语| 午夜欧美大尺度福利影院在线看 | 亚洲一区二区在线视频| 国产精品久久久久9999| 亚洲与欧洲av电影| 狂野欧美激情性xxxx| 亚洲欧洲另类国产综合| 欧美精品久久一区二区| 一本色道久久综合| 久久视频在线看| 日韩亚洲欧美在线观看| 国产精品色在线| 欧美一级视频| 91久久久在线| 久久久久国产一区二区三区| 亚洲国产欧美不卡在线观看 | 久久久不卡网国产精品一区| 欧美第一黄色网| 一区二区三区高清在线观看| 国产精品久久久久aaaa九色| 久久久久久久97| 亚洲免费电影在线| 久久这里只有| 亚洲午夜激情| 亚洲国产精品激情在线观看| 国产精品h在线观看| 久久久美女艺术照精彩视频福利播放| 香蕉久久夜色精品| 亚洲日本黄色| 国产亚洲欧美色| 欧美日韩在线播放一区二区| 欧美一区二区观看视频| 亚洲国产一区二区三区青草影视| 亚洲高清一二三区| 国产精品网站在线| 欧美日本在线一区| 久久综合九色九九| 亚洲欧美一区二区激情| 亚洲激情网站| 欧美mv日韩mv国产网站| 欧美在线视频日韩| 中文国产成人精品| 亚洲精品乱码久久久久久日本蜜臀| 免费人成网站在线观看欧美高清| 美女脱光内衣内裤视频久久影院 | 国产精品久久看| 欧美成人精品1314www| 欧美怡红院视频| 一区二区三区欧美在线| 亚洲黄色天堂| 欧美国产91| 欧美成人精品激情在线观看| 久久精品人人做人人综合| 亚洲已满18点击进入久久| 日韩视频亚洲视频| 亚洲理伦在线| 亚洲国产一区二区三区青草影视| 欧美激情一区二区三区成人| 久久亚洲精品欧美| 久久久久**毛片大全| 欧美中文字幕视频在线观看| 亚洲男人第一网站| 久久精品99国产精品| 性色av一区二区三区在线观看| 国产最新精品精品你懂的| 国产精品一区二区女厕厕| 国产精品亚洲综合色区韩国| 欧美视频在线观看 亚洲欧| 欧美日韩综合久久| 欧美视频一区二区三区| 国产精品扒开腿做爽爽爽视频| 久久激情久久| 久久久www免费人成黑人精品 | 国产亚洲第一区| 国产亚洲精品久久久久婷婷瑜伽| 久久亚洲欧美国产精品乐播| 久久久青草婷婷精品综合日韩| 欧美激情麻豆| 亚洲——在线| 欧美一区视频| 久久综合给合久久狠狠色| 蜜臀av国产精品久久久久| 欧美激情视频给我| 亚洲电影下载| 一本到高清视频免费精品| 亚洲男人的天堂在线aⅴ视频| 欧美激情一二区| 亚洲国产第一页| 一本色道久久综合亚洲精品婷婷 | 亚洲欧美文学| 久久精品论坛| 亚洲国产91精品在线观看| 亚洲精选视频在线| 亚洲女人天堂av| 久久精品国产综合| 欧美成人午夜77777| 欧美日韩综合不卡| 黑人巨大精品欧美一区二区| 91久久久久久| 欧美影院在线| 亚洲国产精品一区二区久| 亚洲调教视频在线观看| 久久免费视频网| 欧美视频四区| 亚洲国产日韩一区| 性8sex亚洲区入口| 欧美激情一区二区久久久| 国产精品99久久久久久人| 久久久中精品2020中文| 欧美日韩在线免费| 亚洲大片在线观看| 亚洲免费一在线| 亚洲第一偷拍| 欧美主播一区二区三区美女 久久精品人| 这里只有视频精品| 久久综合九色欧美综合狠狠| 欧美视频不卡中文| 1769国内精品视频在线播放| 亚洲欧美日韩人成在线播放| 欧美国产精品人人做人人爱| 亚洲无线一线二线三线区别av| 一区二区三区导航| 欧美gay视频| 国模私拍视频一区| 亚洲一级电影| 亚洲大片av| 久久亚洲美女| 黑人巨大精品欧美一区二区| 亚洲女优在线| 日韩一区二区免费看| 美女网站在线免费欧美精品| 国产亚洲精品久久久久久| 西瓜成人精品人成网站| 亚洲精品视频在线观看网站| 久久久亚洲国产天美传媒修理工 | 牛人盗摄一区二区三区视频| 国产女同一区二区| 亚洲综合日韩在线| 亚洲日本激情| 99精品国产99久久久久久福利| 一区二区三区高清| 欧美精品激情在线| 亚洲人成7777| 欧美成人黄色小视频| 久久精品亚洲一区二区三区浴池| 久久在线视频| 狠狠色狠狠色综合人人| 久久狠狠久久综合桃花| 亚洲欧美美女| 国产美女扒开尿口久久久| 亚洲天堂男人| 久久久久久高潮国产精品视| 午夜精品久久99蜜桃的功能介绍| 久久国产一区二区三区| 国产一区二区中文字幕免费看| 亚洲国产精品久久久久| 久久色在线播放| 久久久久久夜|