DELPHI技术

博客园 首页 新随笔 联系 订阅 管理

 类和类成员概述
类(Class),是一个包含字段(Field,也称为域)、方法(Method)和属性(Property)(事件(Event)是一种特殊的属性)三种成员的构造体。

因为本书是讲"Delphi精要",所以对于面向对象理论中的类的概念,就不再使用什么"禽兽 | 家禽 | 鸡鸭鹅"之类例子来讲解了,如果大家对类的基本概念还不是很理解,那么可以参阅相关资料和书籍...
类和类成员概述
类(Class),是一个包含字段(Field,也称为域)、方法(Method)和属性(Property)(事件(Event)是一种特殊的属性)三种成员的构造体。

因为本书是讲"Delphi精要",所以对于面向对象理论中的类的概念,就不再使用什么"禽兽 | 家禽 | 鸡鸭鹅"之类例子来讲解了,如果大家对类的基本概念还不是很理解,那么可以参阅相关资料和书籍。

对象即类的实例,是使用构造函数(在Object Pascal中是用关键字constructors标识的,它是一个特殊的类方法,通常是Create)来生成的一个内存块。销毁一个对象使用析构函数(用关键字destructors标识,通常是Destroy)。

字段,就是在对象中对应某项数据的变量。有的资料和书籍也将字段称为域,在本书中,为了便于理解,一般都称作字段。

而方法则是一些函数和过程。方法可以分为普通方法和类方法两种,分别用来操作对象和类。普通方法只有由类实例调用,而类方法可以由类或者类实例调用。

属性,实际上是一些需要特殊处理的字段的包装,它们的值可以用字段或者方法来存取。

以下是摘自Forms单元的一段代码,这段代码声明了一个TCustomForm类(它是TForm的父类):

 

{TCustomForm派生于另一个类TScrollingWinControl,它们构成父子关系}

TCustomForm = class(TScrollingWinControl)

private

{声明一个字段FWindowState}

FWindowState: TWindowState;

{再声明一个字段FOnDestroy}

FOnDestroy: TNotifyEvent;

......

{声明一个方法SetWindowState}

procedure SetWindowState(Value: TWindowState);

......

public

{这是一个构造函数:Create}

constructor Create(AOwner: TComponent); override;

{这是一个析构函数:Destroy}

destructor Destroy; override;

{声明一个属性WindowState,它从字段FWindowState读取值,用方法SetWindowState
保存值(方法SetWindowState在内部将值保存到字段FWindowState)}

property WindowState: TWindowState read FWindowState write SetWindowState;

{声明一个特殊的属性——事件OnDestroy,和WindowState不同,OnDestroy的存取都是
通过字段FOnDestroy进行的}

property OnDestroy: TNotifyEvent read FOnDestroy write FOnDestroy

......

end;

 

在本小节最后,我们渴望搞清楚类成员的可见性问题。对于类成员的其他深入知识,我会开辟专门的小节来阐述。

类成员的可见性是对该类的使用者而言。在声明一个类时,类可以被分为5个区域,用以下5个关键字标识:

private, protected, public, published, automated。

所有的类成员都被放置在不同的区域里,不同区域的类成员具有不同的可见性。如果类的定义和类的使用者在同一个单元内,那么该类的所有成员无论位于哪个区域,对于使用者而言都是可见的。一个类对于相同单元的其他类来说,类似于C++中的"友类",其所有成员都可以被访问。因此,类成员的可见性设置只是在它们位于不同单元时,才是有效的。这时候,区域内成员的可见性规定如下:

(1) private域:总不可见。这个区域用来隐藏一些实现细节并防止使用者直接修改敏感信息,比如容纳属性的存取字段和方法。

(2) protected:派生类可见。这样既可以起到private域的作用,也能给派生类提供较大的灵活性。该区域常被用来定义虚方法。

(3) public:总可见。通常用来放置构造、析构函数等供使用者调用的方法。

(4) published:总可见。而且这个区域的类成员有运行时类型信息,该区域通常用来放置供使用者访问的属性和事件。

(5) automated:总可见。而且该域的成员具有自动化类型信息,用于创建自动化服务器。该关键字已经不再使用,为向后兼容保留。

类的成员通常都是很明确指定了它所属区域的,但并不总是这样,凡事都是有例外的。比如我们在窗体上放置一个按钮并双击它生成OnClick事件过程后,单元的源代码中对窗体类的定义就变成了下面的样子:

 

type

TForm1 = class(TForm)

Button1: TButton;

procedure Button1Click(Sender: TObject);

private

{ Private declarations }

public

{ Public declarations }

end;

 

我们发现字段Button1和过程Button1Click并没有被明确地放到哪个可见性区域中。那么这时候它们的可见性按什么规则来确定呢?此时和编译指令$M密切相关。

后面我们要讲:$M用来控制编译器是否给类生成运行时类型信息。所以,在{$M+}状态,Button1和Button1Click被隐含指定到published域;在{$M-}状态,则到public域。

那么对于上面的TForm1来说,因为它现在处在{$M+}状态,所以Button1和Button1Click实际上被隐含指定到published域。


深入认识方法
在上一小节,我们已经简单说明了类的成员之一——方法。但是,对于方法仅仅只这么一点认识是远远不够的。

在本小节里,我打算从不同角度对方法分类研究,借以深入认识方法。

(1) 从调用者角度可分为:

① 普通方法;

② 类方法。

普通方法只能被类实例(即对象)调用,而类方法不但可以被对象调用,还可以直接被类调用(比如构造函数Create和析构函数Destroy)。我们看下面的例子:

 

unit Unit1;

 

interface

 

uses

Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls,
Forms, Dialogs, StdCtrls;

 

type

TForm1 = class(TForm)

Button1: TButton;

procedure Button1Click(Sender: TObject);

private

{ Private declarations }

public

{ Public declarations }

end;

 

TOneObject = class

{声明一个类方法ClassProc。方法是在最前面加"Class"关键字}

class procedure ClassProc;

{声明一个普通方法OneProc}

procedure OneProc;

end;

 

var

Form1: TForm1;

 

implementation

 

{$R *.dfm}

 

class procedure TOneObject.ClassProc;

begin

ShowMessage('ClassProc');

end;

 

procedure TOneObject.OneProc;

begin

ShowMessage('OneProc');

end;

 

procedure TForm1.Button1Click(Sender: TObject);

var

OneObj: TOneObject;

begin

TOneObject.ClassProc; {类方法可以直接被类调用}

OneObj := TOneObject.Create; {Create本身也是个类方法}

OneObj. ClassProc; {对象也可以调用类方法}

OneObj.OneProc; {普通方法只能被对象调用}

OneObj.Free;

end;

 

end.

 

类方法是从C++的static(静态)函数借鉴而来的。实现一个类方法时,要特别注意不要让它依赖于任何实例信息,千万不要在类方法中存取字段、属性和普通方法。否则通过类而不是对象来调用它时,将发生错误,因为此时并没有实例信息。

在本小节接下来的内容里,我们不再讨论类方法,所有的方法都是指普通方法。

(2) 从调用机制上分:

① 静态方法。如下面的代码定义了TOneObject的静态方法OneProc。

 

TOneObject = class

procedure OneProc;

end;

 

没有修饰字的方法被默认为静态方法。和下面要讲的虚方法相比,静态方法能够获得更快的运行速度,因为它的地址是编译时确定、运行时映射的;而虚方法为了实现某些更加高级、灵活、复杂的功能,需要在运行时作一些附加处理(比如动态寻址),所以调用时相对要慢一些。

虚方法。虚方法使用关键字virtual或者dynamic声明,如:

 

TOneObject = class

procedure OneProc; virtual;

function OneFun: Boolean; dynamic;

end;

 

其中OneProc和OneFun都是虚方法。虚方法可以在子类中进行覆盖,从而增强方法的功能。覆盖一个虚方法应该使用override关键字。例如我们定义TOneObject的子类:

在子类中,可以(但不是必须的)覆盖父类的虚方法,从而实现更加复杂的控制。覆盖采用关键字override:

 

unit Unit1;

 

interface

 

uses

Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls,
Forms, Dialogs, StdCtrls;

 

type

TForm1 = class(TForm)

Button1: TButton;

procedure Button1Click(Sender: TObject);

private

{ Private declarations }

public

{ Public declarations }

end;

 

{TParent声明了两个虚方法}

TParent = class

procedure OneProc; virtual;

function OneFun: Boolean; dynamic;

end;

 

{TChild派生于TParent,并对父类的两个虚拟方法都做了覆盖}

TChild = class(TParent)

procedure OneProc; override;

function OneFun: Boolean; override;

end;

 

var

Form1: TForm1;

 

implementation

 

{$R *.dfm}

 

{ TParent }

 

procedure TParent.OneProc;

begin

ShowMessage('TParent');

end;

 

function TParent.OneFun: Boolean;

begin

Result := False;

end;

 

{ TChild }

 

procedure TChild.OneProc;

begin

inherited; {inherited调用父类的OneProc的代码,这句的结果是显示'TParent'}

ShowMessage('TChild');

end;

 

function TChild.OneFun: Boolean;

begin

Result := inherited OneFun; {调用父类的OneFun代码}

if not Result then

Result := TRue;

end;

 

procedure TForm1.Button1Click(Sender: TObject);

var

Child: TChild;

begin

Child := TChild.Create;

Child.OneProc; {会先显示'TParent'(父类代码实现的),再显示'TChild'
(子类实现的)}

if Child.OneFun then {条件语句成立}

ShowMessage('OneFun return true');

Child.Free;

end;

end.


使用不同关键字声明的虚方法是有区别的。virtual声明的称为虚拟方法,dynamic声明的称为动态方法。它们在被调用时,派遣机制有所不同,具体可以参看第5章的"虚拟方法表和动态方法表"小节。

因为声明虚方法的目的都是供子类覆盖,所以虚方法一般应该声明在protected区域,当然不是绝对的。如果希望类的使用者能够调用这个虚拟方法,那么还可以声明在public域。

还得对inherited作点说明。inherited并不仅仅局限在子类覆盖后的虚方法中调用父类中被覆盖的方法,实际上,inherited可以使用在任何地方、调用子类可见的任何父类方法(包括protected、public、published等域的)。

③ 抽象方法。它是虚方法的特例,在虚方法声明后加上abstract关键字构成,如:

 

TParent = class

procedure OneProc; virtual; abstract;

function OneFun: Boolean; dynamic; abstract;

end;

 

抽象方法和普通虚方法的区别:

a. 抽象方法只有声明,没有实现;而虚方法必须有实现部分,哪怕没有实际代码而只有begin...end头。

b. 抽象方法必须在子类中覆盖并实现后才用调用。因为没有实现的方法不能被分配实际地址,而调用一个没有实际地址的方法显然是荒谬的。

所以,抽象方法也可以被称为纯虚方法。

如果一个类中含有抽象方法,那么这个类就成了抽象类,如TStrings含有:

procedure Clear; virtual; abstract;

procedure Delete(Index: Integer); virtual; abstract;

等多个抽象方法。

抽象类是不应该直接用来创建实例的,因为一旦调用了抽象方法,将抛出地址异常,而我们很难保证不在某些地方调用抽象方法。所以,尽管实例化抽象类是被允许的,却是应该避免的。

因此,抽象类一般都是中间类,实际使用的总是覆盖实现了抽象方法的子类。比如常用的字符串列表类TStrings,我们总是使用它的子类而不是它本身来构造实例,如:

 

var

Strs: TStrings;

begin

Strs := TStringList.Create;

......

end;

(3) 从用途来分:

① 重载方法。方法名相同,但参数个数或者类型不同的多个方法构成重载;重载的目的是得到多个同名但是功能不同的方法。重载是用关键字overload来指明的,比如:

 

TParent = class

procedure OneProc; overload;

function OneProc(S: String): Boolean; overload;

end;

 

上面TParent类的方法OneProc被重载。

重载方法的几个特点:

a. 可以分别是函数或者过程。因为在Delphi中,可以将过程看做一个没有返回值的函数,一个函数也可以当作过程调用。

b. 如果位于相同类中,都必须加上overload关键字;如果分别在父类和子类中,那么父类的方法可不加overload而子类必须加overload。

c. 如果父类的方法是虚(virtual或者dynamic)的,那么在子类中重载该方法时应该加上reintroduce修饰字,否则会出现编译警告:"hides virtual method of base type"。当然只是编译时产生警告,如果你不顾它的警告,坚持不加修饰字,对程序运行结果也不会造成影响。如:

 

TParent = class

procedure OneProc; virtual;

end;

 

TChild = class(TParent)

procedure OneProc; reintroduce; overload;

end;

 

d. 在published区不能出现多个相同的重载方法。如:

 

TParent = class

procedure OneProc; virtual;

end;

 

TChild = class(TParent)

published

procedure OneProc; reintroduce; overload;

{和父类构成方法重载关系是可以的,因为在TChild的published区,只有一个OneProc方
法,而下面两行企图重载AnotherProc则是没可能的,编译器不允许在published区出现
多个同名的方法}

procedure AnotherProc(S: String); overload;

procedure AnotherProc; overload;

end;

为什么编译器不允许在published出现多个同名的方法呢?别忘了在前面我们说过:published区的类成员会生成运行时类型信息的,而类成员是通过名字区分的。因此,这时候编译器无法为成员AnotherProc生成运行时类型信息。
程序运行时,能够根据传入的参数来正确区分应该调用哪一个被重载的方法。

重载的概念对于普通的过程和函数也是适用的,实际上方法重载是从普通过程和函数的重载引申而来的。我之所以没有将重载内容放在3.2节,是因为方法重载更加复杂,在这里可以更加全面地来阐述它。

posted on 2005-07-22 14:14  DELPHI技术  阅读(797)  评论(0编辑  收藏  举报