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

積木

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 閱讀(258) 評論(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>
            麻豆av一区二区三区| 亚洲国产精品成人综合| 国产精品日韩欧美综合| 久久久久久91香蕉国产| 99在线精品观看| 麻豆av福利av久久av| 亚洲欧美一区二区原创| 亚洲人成网站影音先锋播放| 国产精品嫩草99a| 欧美伦理影院| 免费91麻豆精品国产自产在线观看| 亚洲免费伊人电影在线观看av| 亚洲日韩中文字幕在线播放| 欧美顶级少妇做爰| 狠狠久久五月精品中文字幕| 国产精品地址| 欧美伦理影院| 欧美久久久久久蜜桃| 快射av在线播放一区| 欧美日本在线播放| 欧美在线免费观看亚洲| 午夜精品在线看| 亚洲一级在线| 在线一区二区三区四区| 日韩视频中文| 亚洲免费高清| 99精品视频免费观看| 亚洲精品久久久久久下一站 | 欧美大片免费久久精品三p | 午夜亚洲福利在线老司机| 夜夜嗨av一区二区三区四季av| 91久久精品国产91性色tv| 欧美国产日韩亚洲一区| 欧美承认网站| 欧美大片在线观看| 欧美高清不卡在线| 亚洲成人在线视频播放 | 久久爱另类一区二区小说| 午夜精品久久久久影视| 先锋影院在线亚洲| 性做久久久久久免费观看欧美| 香蕉精品999视频一区二区 | 亚洲国内精品| 亚洲美女一区| 一区二区三区欧美在线| 亚洲婷婷在线| 午夜精品剧场| 久久精品123| 免费影视亚洲| 乱中年女人伦av一区二区| 免费看成人av| 欧美激情一区三区| 国产精品日韩在线观看| 亚洲欧美另类国产| 欧美在线黄色| 美国十次成人| 欧美大片一区二区三区| 欧美日韩国产在线一区| 欧美性色视频在线| 国产视频在线观看一区二区三区| 国产手机视频一区二区| 在线观看一区| 日韩午夜电影在线观看| 亚洲综合日韩在线| 久久久.com| 亚洲电影在线| 夜色激情一区二区| 亚洲欧美综合国产精品一区| 欧美一区二区三区日韩视频| 久久先锋影音av| 欧美日韩三级| 国产亚洲欧美一级| 亚洲精品国产系列| 午夜激情综合网| 美日韩在线观看| 99视频热这里只有精品免费| 欧美一区二区三区视频免费播放| 久久久夜色精品亚洲| 欧美日韩一区二区三区在线视频 | 国产一区二区三区四区hd| 亚洲国产精品一区二区尤物区 | 影音先锋欧美精品| 野花国产精品入口| 久久免费视频观看| 亚洲美女黄网| 欧美一区二区三区免费观看视频 | 欧美精品免费在线| 国产一区二区三区的电影| 日韩亚洲视频| 久久av一区二区三区| 亚洲国产乱码最新视频| 亚洲欧美激情在线视频| 欧美二区在线观看| 国产亚洲欧美一区| 亚洲一区久久久| 免费观看在线综合| 亚洲欧美日本国产专区一区| 欧美大尺度在线| 亚洲第一视频| 性欧美精品高清| 亚洲国产精品精华液2区45| 亚洲综合成人婷婷小说| 久久久之久亚州精品露出| 欧美午夜剧场| 亚洲精品一区二区三区樱花| 久久久免费av| 亚洲一区精品视频| 欧美日本视频在线| 亚洲国产成人精品久久久国产成人一区 | 欧美一级艳片视频免费观看| 欧美精品免费观看二区| 在线日韩av永久免费观看| 性欧美xxxx视频在线观看| 亚洲国产裸拍裸体视频在线观看乱了| 久久爱另类一区二区小说| 国产伦理一区| 亚洲一区三区电影在线观看| 欧美激情aⅴ一区二区三区| 久久不见久久见免费视频1| 国产精品视频免费一区| 中日韩美女免费视频网址在线观看 | 亚洲无亚洲人成网站77777| 欧美激情亚洲自拍| 亚洲电影在线观看| 狂野欧美激情性xxxx欧美| 亚洲欧美日韩国产中文在线| 欧美三级电影网| 一本色道久久88综合日韩精品| 欧美成ee人免费视频| 久久久久久日产精品| 国语自产偷拍精品视频偷| 久久精品国产99| 欧美亚洲视频一区二区| 国产人成精品一区二区三| 欧美一二三区精品| 亚洲——在线| 国产精品一区免费观看| 亚洲欧美国产毛片在线| 一区二区三区日韩| 欧美午夜免费电影| av成人毛片| 日韩午夜在线播放| 欧美日韩一区成人| 亚洲午夜一级| 亚洲视频综合| 国产精品亚洲综合久久| 香蕉成人啪国产精品视频综合网| 麻豆九一精品爱看视频在线观看免费| 在线不卡a资源高清| 欧美一区久久| 亚洲高清视频中文字幕| 麻豆久久婷婷| 最新中文字幕亚洲| 亚洲人午夜精品| 欧美连裤袜在线视频| 夜夜嗨av一区二区三区网站四季av | 久久久精品性| 亚洲男人第一网站| 一区二区欧美在线| 国产精品第三页| 香蕉免费一区二区三区在线观看| 亚洲一区二区三区成人在线视频精品| 国产精品一区2区| 久久久综合香蕉尹人综合网| 久久在线91| 艳女tv在线观看国产一区| 亚洲视频一区二区| 国产精品一区久久| 另类尿喷潮videofree| 欧美jizzhd精品欧美喷水 | 999亚洲国产精| 亚洲图片欧美日产| 国内成人精品视频| 亚洲第一福利视频| 国产精品hd| 久久久久久久久久看片| 蜜臀va亚洲va欧美va天堂| 一本色道久久88综合亚洲精品ⅰ| 亚洲一区二区三区免费在线观看| 国产一区香蕉久久| 欧美成年人在线观看| 欧美日韩小视频| 久久精品91| 欧美成人免费在线| 香蕉乱码成人久久天堂爱免费 | 亚洲黄色在线观看| 国产精品久久久久久久免费软件| 久久久久久久网| 欧美精品在线观看91| 久久国产99| 欧美韩国日本综合| 欧美日韩成人一区| 国产精品久久久久久久久久尿 | 91久久久亚洲精品| 在线一区免费观看| 国内精品久久久久久久果冻传媒 | 欧美一区二区三区婷婷月色| 亚洲激情在线观看| 亚洲一区综合| 亚洲精品欧美一区二区三区|