在今后的 Delphi 中, 以接口、結構為主的設計應該會越來越多, 因為這樣太方便了.
System.RegularExpressions 就是以結構為主體設計的非常好的示范; 但更多東西使用接口會更合適.
有見過他人早就使用接口寫程序, 從手頭的這個程序開始我才開始使用.
現在基本總結出四種框架模式: 1、直接實現; 2、間接實現(或叫繼承實現); 3、覆蓋實現; 4、委托實現.
一、直接實現:
下例中雖有 TMy1、TMy2, 但在具體應用中使用的應是 IMy1、IMy2, 這就是我所謂的以接口為主導.
TMy1、TMy2 直接實現了所屬接口的所有方法, 這是我所謂的直接實現.
這樣可能會有代碼重復, 但如果程序很小, 還是挺實用的.
unit Unit1; interface uses Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs; type TForm1 = class(TForm) procedure FormCreate(Sender: TObject); end; IA = Interface procedure Method_A; end; IB = Interface(IA) procedure Method_B; end; IMy1 = Interface(IB) procedure Method_My1; end; IMy2 = Interface(IB) procedure Method_My2; end; TMy1 = class(TInterfacedObject, IMy1) procedure Method_A; procedure Method_B; procedure Method_My1; end; TMy2 = class(TInterfacedObject, IMy2) procedure Method_A; procedure Method_B; procedure Method_My2; end; var Form1: TForm1; implementation {$R *.dfm} { TMy1 } procedure TMy1.Method_A; begin ShowMessage('A'); end; procedure TMy1.Method_B; begin ShowMessage('B'); end; procedure TMy1.Method_My1; begin ShowMessage('My1'); end; { TMy2 } procedure TMy2.Method_A; begin ShowMessage('A'); end; procedure TMy2.Method_B; begin ShowMessage('B'); end; procedure TMy2.Method_My2; begin ShowMessage('My2'); end; {測試} procedure TForm1.FormCreate(Sender: TObject); var v1: IMy1; v2: IMy2; begin v1 := TMy1.Create; v1.Method_A; v1.Method_B; v1.Method_My1; v2 := TMy2.Create; v2.Method_A; v2.Method_B; v2.Method_My2; end; end.
二、間接實現:
下面例子通過一個間接的 TB 類, 避免了 TMy1、TMy2 中可能會重復的代碼.
unit Unit1; interface uses Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs; type TForm1 = class(TForm) procedure FormCreate(Sender: TObject); end; IA = Interface procedure Method_A; end; IB = Interface(IA) procedure Method_B; end; TB = class(TInterfacedObject, IB) procedure Method_A; procedure Method_B; end; IMy1 = Interface(IB) procedure Method_My1; end; IMy2 = Interface(IB) procedure Method_My2; end; TMy1 = class(TB, IMy1) procedure Method_My1; end; TMy2 = class(TB, IMy2) procedure Method_My2; end; var Form1: TForm1; implementation {$R *.dfm} { TB } procedure TB.Method_A; begin ShowMessage('A'); end; procedure TB.Method_B; begin ShowMessage('B'); end; { TMy1 } procedure TMy1.Method_My1; begin ShowMessage('My1'); end; { TMy2 } procedure TMy2.Method_My2; begin ShowMessage('My2'); end; {測試} procedure TForm1.FormCreate(Sender: TObject); var v1: IMy1; v2: IMy2; begin v1 := TMy1.Create; v1.Method_A; v1.Method_B; v1.Method_My1; v2 := TMy2.Create; v2.Method_A; v2.Method_B; v2.Method_My2; end; end.
三、覆蓋實現:
從 TB 繼承的過程中當然也可以通過覆蓋虛函數而實現多態, 下面的 TMy2 就這么做了.
unit Unit1; interface uses Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs; type TForm1 = class(TForm) procedure FormCreate(Sender: TObject); end; IA = Interface procedure Method_A; end; IB = Interface(IA) procedure Method_B; end; TB = class(TInterfacedObject, IB) procedure Method_A; virtual; procedure Method_B; virtual; end; IMy1 = Interface(IB) procedure Method_My1; end; IMy2 = Interface(IB) procedure Method_My2; end; TMy1 = class(TB, IMy1) procedure Method_My1; end; TMy2 = class(TB, IMy2) procedure Method_A; override; procedure Method_B; override; procedure Method_My2; end; var Form1: TForm1; implementation {$R *.dfm} { TB } procedure TB.Method_A; begin ShowMessage('A'); end; procedure TB.Method_B; begin ShowMessage('B'); end; { TMy1 } procedure TMy1.Method_My1; begin ShowMessage('My1'); end; { TMy2 } procedure TMy2.Method_A; begin ShowMessage('A_My2'); end; procedure TMy2.Method_B; begin ShowMessage('B_My2'); end; procedure TMy2.Method_My2; begin ShowMessage('My2'); end; {測試} procedure TForm1.FormCreate(Sender: TObject); var v1: IMy1; v2: IMy2; begin v1 := TMy1.Create; v1.Method_A; v1.Method_B; v1.Method_My1; v2 := TMy2.Create; v2.Method_A; v2.Method_B; v2.Method_My2; end; end.
四、委托實現:
接口中的方法是肯定要實現的, 但也可以通過 implements 關鍵字借用(或叫委托)其它的實現;
但, 官方文檔說這只適用於 Win32. 就是說這種方法在 Win64 和其它系統都不行, 還學它干嘛?