在今后的 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 和其它系統都不行, 還學它干嘛?
