程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> Delphi >> Visitor模式

Visitor模式

編輯:Delphi

目的

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

 [Gam+, page 331].

動機

考慮一個面向對象的建模工具,比如說‘Rational Rose、ModelMaker’,它將一個模型表示為類和類的成員。

在建模工具上提供了許多操作成員功能,比如:列表類的所有成員、生成類的代碼框架、反向工程等。

這些操作大多對不同的成員進行不同的操作。它將成員分成字段(fIElds)、方法(methods)、

屬性(propertIEs)。因些我們必須建立專門處理字段的類,專門處理methods的類等等。成員類的集合當然依賴被編譯的語言。但對於一給定語言變化不大。

 

<?XML:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" /> 

 

如圖顯示了部分成員類的框架。問題產生了,如果我將所有這些操作分散到不同的成員類,

將會導致整個系統難於理解,修改,維護。將類代碼生成與類成員檢查放在一起,將產生混亂。些外加入新的操作時要重新編譯的有的類(至少也重新編譯所有的相關的系)。有個辦法:你可能獨立的增加一個新的操作,並這個成員類獨立如作用於其上的操作。

要實現上述兩個目標,我們可以將每個類中相關操作包裝在一上獨立的對象(稱為visitor

並在遍歷類成員列表時將此對象傳遞給當前成員。當一個成員‘接受’ 訪問,該成員向訪問者發送包含自身信息的請求。該成員請自本身作為一個參數。訪問者執行這些操作。

例如:一個不使用訪問者的代碼生成器可能會通成員類的抽象的方法:TMember.WriteInterfaceCode(Output: TStream)生成代碼。每一個成員都會調用WriteInterfaceCode生成適當的輸出代碼。如果通過訪問者來生成代碼,則會創建一個TinterfaceCodeVisitor對象,並在成員列表上調用參數為訪問對象的AcceptVisitor方法。每一個在員在實現AcceptVisitor將會回調visitor一個字段將調用訪問者的VisitFIEld方法,而一個方法則調用VisitMethod方法。這樣,以前類TfIEld的WriteInterfaceCode操作現在成為TinterfaceCodeVisitor的VisitFIEld操作。

 

為使訪問者不僅僅只做代碼生成,我們需要所有的成員列表的訪問者有一個抽象的父類TmemberVisitor。TmemberVisitor必須為每一個成員定義一種方法。一個需要將成員輸出為Html格式的應用將定義TmemberVisitor新的子類,並不再需要在成員類中增加與特定應用相關的代碼。Visitor模式將每個操作封裝在一個相關的Visitor中

 

  
 

 

 

使用Visitor模式,必須定義兩個層次的類:一個應於接受操作的元素(Tmember層次)另一個定義於對元素的操作(TmemberVisitor 層次)。增加一個新的操作時只需給訪問者層次增加一個新的子類。我可能簡單的定義新的TmemberVisitor子類以增加新的功能。

 

應用

下面的代碼演示上面描述的類Tmember的Visitor模式的應用

 

type

  TMember = class (TObject)

  public

    procedure AcceptMemberVisitor(Visitor: TMemberVisitor); virtual;

  end;

 

  TFIEld = class (TMember)

  public

    procedure AcceptMemberVisitor(Visitor: TMemberVisitor); override;

  end;

 

  TMethod = class (TMember)

  public

    procedure AcceptMemberVisitor(Visitor: TMemberVisitor); override;

  end;

 

  TProperty = class (TMember)

  public

    procedure AcceptMemberVisitor(Visitor: TMemberVisitor); override;

  end;

 

  TMemberVisitor = class (TObject)

  public

    procedure VisitField(Instance: TFIEld); virtual;

    procedure VisitMember(Instance: TMember); virtual;

    procedure VisitMethod(Instance: TMethod); virtual;

    procedure VisitProperty(Instance: TProperty); virtual;

  end;

 

implementation

 

{ TMember }

 

begin

  Visitor.VisitMember(Self);

end;

 

{ TFIEld }

procedure TFIEld.AcceptMemberVisitor(Visitor: TMemberVisitor);

begin

end;

 

{ TMethod }

procedure TMethod.AcceptMemberVisitor(Visitor: TMemberVisitor);

begin

  Visitor.VisitMethod(Self);

end;

 

{ TProperty }

procedure TProperty.AcceptMemberVisitor(Visitor: TMemberVisitor);

begin

  Visitor.VisitProperty(Self);

end;

 

{ TMemberVisitor }

procedure TMemberVisitor.VisitField(Instance: TFIEld);

begin

end;

 

procedure TMemberVisitor.VisitMember(Instance: TMember);

begin

end;

 

procedure TMemberVisitor.VisitMethod(Instance: TMethod);

begin

end;

 

procedure TMemberVisitor.VisitProperty(Instance: TProperty);

begin

end;

 

說明:

?      TMember, TFIEld, TMethod 和 Tproperty都實現了AcceptMemberVisitor方法. 這些方法都嵌入模式中

?      TMemberVisitor 類實現了VisitMember, VisitFIEld等方法。TmemberVisitor是一個抽象的類,它所有的方法由具體的子類實現。

下面是一個簡單的代碼生成器的實現。

代碼介紹:

?      TCodeGenerationVisitor 是一個用於實現成員的代碼生成器的訪問者。

?      訪問者定義了一個上下文相關的屬性:Output: TTextStream,

?       它必須在VisitXXX調用前被定,如:DrawingVisitor典型的需要一個包括canvas的上下文,來支持畫圖操作。上下文在遍歷整個member對列前賦予了代碼生成器。

?      代碼生成器將整結的生成的類的所有代碼

 

要真正的了解Visitor模式,你可執行這個例子 ,並進一步的學習雙分派機制: accept/visit.

 

unit CodeGenerators;

 

interface

 

uses Classes, TextStreams;

 

type

 

  TCodeGenerator = class (TObject)

  public

    procedure Generate(Members: TList; Output: TTextStream);

  end;

 

implementation

 

uses Members;

 

type

  TCodeGenerationVisitor = class (TMemberVisitor)

  private

    FOutput: TTextStream;

  public

    procedure VisitField(Instance: TFIEld); override;

    procedure VisitMethod(Instance: TMethod); override;

    procedure VisitProperty(Instance: TProperty); override;

    property Output: TTextStream read FOutput write FOutput;

  end;

 

 

{ TCodeGenerationVisitor }

procedure TCodeGenerationVisitor.VisitField(Instance: TFIEld);

begin

  Output.WriteLnFmt('  %s: %s;', [Instance.Name, Instance.DataName]);

end;

 

procedure TCodeGenerationVisitor.VisitMethod(Instance: TMethod);

var

  MKStr, DTStr: string;

begin

  case Instance.MethodKind of

    mkConstructor: MKStr := 'constructor';

    mkDestructor: MKStr := 'destructor';

    mkProcedure: MKStr := 'procedure';

    mkFuntion: MKStr := 'function';

  end;

  if Instance.MethodKind = mkFunction then

    DTStr := ': ' + Instance.DataName

  else

    DTStr := ';

  {代碼不完整,現足以演示Tmethod代碼生成 }

  Output.WriteLnFmt('  %s %s%s%s;'

                    [MKStr, Instance.Name, Instance.Parameters, DTStr]);

end;

 

procedure TCodeGenerationVisitor.VisitProperty(Instance: TProperty);

begin

  Output.WriteLnFmt('  property %s: %s read %s write %s;',

                    [Instance.Name, Instance.DataName,

                     Instance.ReadSpecifier, Instance.WriteSpecifIEr]);

end;

 

{ TCodeGenerator }

procedure TCodeGenerator.Generate(Members: TList; Output: TTextStream);

var

  I: Integer;

begin

  {寫入類定義 }

  Output.WriteLine('TSample = class (TObject)');

 

 

  {好! 加入代碼生成器的訪問者}

  Visitor := TCodeGenerationVisitor.Create;

  Try

    {記住為訪問都提供上下文,以便更好的訪問VisitXXX方法。}

    for I := 0 to Members.Count - 1 do

      { 代碼的具體段,好事情發生了}

      TMember(Members[I]).AcceptMemberVisitor(Visitor);

  finally

    Visitor.Free;

  end;

  {類成員的代碼生成完畢}

  Output.WriteLine('end;');

end;

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved