delphi 中 virtual、reintroduce、override关键词的使用

虚拟方法和动态方法不同于静态方法,它们可以在其后裔类中被覆盖(overridden),即只有一个方法所在类的祖先类对应的方法是virtual或dynamic的,这个当前类的方法才可以用overide覆盖。当一个覆盖方法被调用时,方法调用中使用的类或对象的实际(运行时)类型决定了哪一个实现是有效的,而非变量声明的类型决定。virtual方法是可以有实现的部分的,也可是在后裔类中override。

abstract抽象方法是那些在类中声明但未实现的虚拟方法或动态方法。抽象方法的实现推延到后裔类中。声明抽象方法必需在指示字virtual或dynamic之后使用abstract。例如:procedure DoSomething; virtual; abstract;  即abstract方法是没有实现的,实现推延到后裔类中,后裔override这个方法后可以有实现部分。


如果:
TBaseObj=class
  public
    procedure AMethod;
  end;

TChildObj=class(TBaseObj)
    procedure AMethod;
  end;

{ TBaseObj }

procedure TBaseObj.AMethod(str:string);
begin
  showmessage('TBaseObj.AMethod;');
end;

{ TObj }

procedure TChildObj.AMethod(str:string);
begin
  showmessage('TChildObj.AMethod;');
end;

实际上两个类的AMethod方法无任何关系。子类中将看不到祖先类的AMethod方法,而只有自已的AMethod方法,运行下面事件过程将执行
 showmessage('TChildObj.AMethod;');

procedure TForm1.Button2Click(Sender: TObject);
var
  childobj:TChildObj;
begin
  childobj:=TChildObj.Create;
  childobj.AMethod;

end;

如果

TBaseObj=class
  public
    procedure AMethod(str:string;i:integer=0);virtual;
  end;

TChildObj=class(TBaseObj)
    procedure AMethod(str:string);
  end;


pprocedure TBaseObj.AMethod(str:string;i:integer=0);
begin
  showmessage('Base'+str);
end;

{ TObj }

procedure TChildObj.AMethod(str:string;i:integer=0);
begin
  showmessage('Child'+str);
end;

执行下面事件过程,将看到showmessage中显示'ChildAMethod;'  并有[Warning] Unit1.pas(28): Method 'AMethod' hides virtual method of base type 'TBaseObj'的编译信息显示,说明在子类没有用override关键字时,声明一个与祖先类同名的方法是会隐藏祖先类的同名方法,执行自已的方法。而在子类方法声明后加reintroduce就不会有编译信息显示。如果使用override,要保证子类与祖先类方法的参数类型,顺序返回值相同,不然会在编译时报错:
因此交上面子类声明改为:procedure AMethod(str:string;i:integer=0);override;,实现部分也要相应更改,这样执行下面事件过程,即子类方法覆盖了祖先类方法最后将看到showmessage中显示'ChildAMethod;' ,如果有多个子类,都override那个virtual的祖先类的方法,那么最后执行结果是子类的这个AMethod方法,覆盖了祖先类同名方法的实现内容。实际上如果祖先类的这个方法是virtual;abstract的结果也是一样的,。即祖先类只定义一个声明,具体实现由这个祖先类的子类们去完成,执行不同子类结果不同,因为各子类的AMethod实现细节都有所不同。

procedure TForm1.Button2Click(Sender: TObject);
var
  childobj:TChildObj;
begin
  childobj:=TChildObj.Create;
  childobj.AMethod('AMethod;');

end;

另一个有用的东西是:inherited关键字,如果将上面的子类方法实现改为:
procedure TChildObj.AMethod(str:string;i:integer=0);
begin
  inherited;
  showmessage('Child'+str);
end;
然后再运行Button2Click,则会先看到showmessage出BaseAMethod,再看到showmessage出ChildAMethod。为什么会有这种现象呢,看一下关于inherited的说明:
  保留字inherited在实现多种行为中扮演特殊的角色。它可以出现在方法定义中,在其后面可以有或没有标识符。
如果inherited之后跟随一个成名名称,那么除了表示在封装了方法的类的直接祖先中搜寻成员之外,还可以表示标准的方法调用或者对属性或域的引用。

也就是说运行到inherited时会告诉当前inherited所在方法中祖先类的同名方法,所以先看到showmessage出BaseAMethod。如果祖先类中有一个abc方法,那么在此也可调用,即 inherited abc,即如果inherited后面没参数,执行时找祖先类同名方法,有参数如abc则在祖先类找那个
abc方法。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个使用Delphi的示例代码,用于更新数据库的数据: unit UpdateForm; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls, ExtCtrls, DB, ADODB; type TfrmUpdate = class(TForm) lblTitle: TLabel; pnlContent: TPanel; lblName: TLabel; edtName: TEdit; lblAge: TLabel; edtAge: TEdit; btnSave: TButton; btnCancel: TButton; lblID: TLabel; edtID: TEdit; qUpdate: TADOQuery; procedure FormClose(Sender: TObject; var Action: TCloseAction); procedure btnCancelClick(Sender: TObject); procedure btnSaveClick(Sender: TObject); private { Private declarations } FConnection: TADOConnection; public { Public declarations } constructor Create(AOwner: TComponent; Connection: TADOConnection; ID: Integer; Name: string; Age: Integer); reintroduce; end; var frmUpdate: TfrmUpdate; implementation {$R *.dfm} constructor TfrmUpdate.Create(AOwner: TComponent; Connection: TADOConnection; ID: Integer; Name: string; Age: Integer); begin inherited Create(AOwner); FConnection := Connection; edtID.Text := IntToStr(ID); edtName.Text := Name; edtAge.Text := IntToStr(Age); end; procedure TfrmUpdate.btnCancelClick(Sender: TObject); begin Close; end; procedure TfrmUpdate.btnSaveClick(Sender: TObject); begin qUpdate := TADOQuery.Create(nil); try qUpdate.Connection := FConnection; qUpdate.SQL.Text := 'UPDATE tblPerson SET Name=:Name, Age=:Age WHERE ID=:ID'; qUpdate.Parameters.ParamByName('Name').Value := edtName.Text; qUpdate.Parameters.ParamByName('Age').Value := StrToIntDef(edtAge.Text, 0); qUpdate.Parameters.ParamByName('ID').Value := StrToIntDef(edtID.Text, 0); qUpdate.ExecSQL; ModalResult := mrOK; finally qUpdate.Free; end; end; procedure TfrmUpdate.FormClose(Sender: TObject; var Action: TCloseAction); begin Action := caFree; end; end. 在这个示例,我们使用了一个ADOQuery组件来更新数据库的数据。该组件使用连接对象FConnection连接到数据库,然后执行SQL语句来更新tblPerson表的数据。我们在构造函数传递了连接对象、ID、名称和年龄参数。在保存按钮的单击事件,我们将这些值与SQL语句的参数进行匹配,并使用ExecSQL方法来执行更新。如果更新成功,则设置ModalResult为mrOK,以便在主窗体处理更新结果。最后,我们使用FormClose事件来释放窗体的资源。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值