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

積木

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

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

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 閱讀(260) 評論(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>
            欧美精品国产精品日韩精品| 欧美成人高清视频| 国产精品久久久久aaaa樱花| 日韩亚洲在线| 亚洲欧洲日本国产| 欧美va亚洲va香蕉在线| 亚洲精品免费在线| 亚洲日本免费| 国产精品一区二区三区乱码| 午夜精品久久久久久久男人的天堂 | 激情久久久久久久久久久久久久久久| 久久激情五月激情| 久久精品综合| 日韩视频永久免费| 一本色道**综合亚洲精品蜜桃冫| 国产精品视频一区二区三区| 久久久7777| 美女久久网站| 亚洲欧美精品| 久久久久久9| 在线视频精品一区| 午夜久久资源| 999在线观看精品免费不卡网站| 一区二区三区偷拍| 激情欧美亚洲| 亚洲性视频h| 亚洲国产精品va在线看黑人动漫| 亚洲伦理在线观看| 国产一区亚洲一区| 亚洲国产高清自拍| 国产精品男女猛烈高潮激情| 美日韩精品免费| 欧美日一区二区三区在线观看国产免| 久久国产一区二区三区| 欧美精品在线免费观看| 久久亚洲精品一区| 欧美日韩综合在线| 欧美高清你懂得| 国产日韩高清一区二区三区在线| 蜜桃伊人久久| 国产精品日本精品| 亚洲欧洲一区二区在线播放| 国外视频精品毛片| 亚洲私人影院| 欧美在线视频一区| 亚洲国产一区二区三区在线播| 一本色道88久久加勒比精品| 亚洲日本中文字幕| 久久国产一二区| 亚洲一区二区三区四区中文| 麻豆久久婷婷| 久久久久国产一区二区三区| 国产精品久久久久久久7电影| 欧美成人69av| 精品成人在线观看| 午夜激情综合网| 午夜亚洲性色视频| 欧美人与性动交a欧美精品| 欧美xxx成人| 国产专区一区| 亚洲欧美三级在线| 欧美一区二区视频免费观看| 国产精品第三页| 亚洲精品你懂的| 精品福利电影| 久久一区国产| 亚洲国产成人porn| 亚洲黄色影片| 欧美不卡一区| 亚洲欧洲一区二区三区久久| 91久久精品国产91性色| 欧美成人午夜剧场免费观看| 欧美国产国产综合| 亚洲欧洲精品一区| 欧美第一黄色网| 亚洲激情网站免费观看| 99精品国产在热久久下载| 欧美精品三级在线观看| 日韩一区二区精品在线观看| 一区二区成人精品 | 欧美在线日韩| 久久久久国产精品一区二区| 国产午夜精品理论片a级探花 | 亚洲深爱激情| 性色av一区二区三区红粉影视| 国产伦一区二区三区色一情| 香蕉久久国产| 男女激情久久| 一本色道久久综合亚洲精品不| 欧美日韩国产大片| 亚洲女优在线| 欧美成人精品影院| 日韩亚洲综合在线| 国产精品视频免费在线观看| 香蕉成人啪国产精品视频综合网| 久久综合久久久久88| 亚洲国产一二三| 国产精品视频大全| 久久精品亚洲国产奇米99| 欧美激情国产日韩| 亚洲综合色婷婷| 在线成人激情视频| 欧美日韩日韩| 久久久久久网站| 一本色道久久综合亚洲精品不卡| 欧美中日韩免费视频| 亚洲国产精品第一区二区| 欧美性猛交视频| 久久久噜噜噜久久中文字免| 亚洲精品一区在线| 蜜桃av一区二区| 亚洲专区在线视频| 亚洲第一搞黄网站| 国产精品网红福利| 男男成人高潮片免费网站| 亚洲一区免费网站| 亚洲国产中文字幕在线观看| 午夜亚洲影视| 99精品99| 亚洲国产精品成人va在线观看| 国产精品亚洲综合一区在线观看| 美女精品在线| 久久久99国产精品免费| 一区二区日韩伦理片| 欧美国产免费| 另类图片综合电影| 欧美在线视频免费| 亚洲香蕉在线观看| 一本色道久久综合狠狠躁篇怎么玩 | 亚洲国产经典视频| 国产女人水真多18毛片18精品视频| 欧美激情四色 | 欧美中文字幕在线播放| 亚洲精品中文字幕女同| 美女视频黄 久久| 久久噜噜亚洲综合| 欧美自拍偷拍| 午夜欧美视频| 亚洲淫片在线视频| 宅男精品视频| 正在播放欧美视频| 一本一本久久a久久精品牛牛影视| 激情六月综合| 韩国三级电影一区二区| 国产情侣久久| 国产一区视频在线观看免费| 国产日韩欧美制服另类| 国产精品视频精品视频| 国产精品久久9| 欧美性事在线| 欧美午夜欧美| 国产精品卡一卡二| 国产精品久久久久久久第一福利 | 久久精品免费观看| 久久久久久精| 免费久久99精品国产| 免费欧美在线视频| 欧美乱妇高清无乱码| 欧美视频免费看| 国产精品剧情在线亚洲| 国产农村妇女毛片精品久久麻豆| 国产精品老女人精品视频| 国产精品久久午夜夜伦鲁鲁| 国产精品久久久久久五月尺| 国产精品亚洲а∨天堂免在线| 国产精品色婷婷久久58| 黑丝一区二区三区| 最近中文字幕日韩精品| 一区二区三区欧美日韩| 午夜精品影院在线观看| 久久视频在线看| 亚洲国产精品va在看黑人| 99精品国产在热久久婷婷| 亚洲一区二区三区涩| 欧美一区二区三区免费视频| 久久久久久久久蜜桃| 欧美v国产在线一区二区三区| 欧美日韩hd| 国产揄拍国内精品对白| 亚洲娇小video精品| 午夜视频一区二区| 欧美h视频在线| 这里只有精品视频在线| 久久综合国产精品| 国产精品第一区| 亚洲福利在线看| 欧美亚洲在线视频| 欧美激情亚洲激情| 亚洲欧美电影在线观看| 狂野欧美激情性xxxx| 国产精品黄色在线观看| 在线观看一区视频| 亚洲一区二区三区在线播放| 另类专区欧美制服同性| 一区二区三区视频在线| 久久一区二区三区四区| 国产精品亚洲综合久久| 一区二区不卡在线视频 午夜欧美不卡'| 久久精品动漫| 一区二区三区视频在线|