码迷,mamicode.com
首页 > Windows程序 > 详细

delphi 基础之二 面向对象概念初步

时间:2015-06-30 14:37:13      阅读:147      评论:0      收藏:0      [点我收藏+]

标签:

面向对象概念初步

1.类的定义

类的定义分两步:首先在类(单元)的接口(interface)部分说明这个方法.然后在实现部分(implementation)部分编写方法的实现代码.

 

定义:

type

类名=class(父类名)

 数据域说明;      //类内部使用变量/常量的声明;

方法说明首部;

end;

 

实现代码:

procedure 类名.方法(参数);

 实现代码;

end;

2.创建对象及对象成员的引用

创建对象分两步:

首先声明对象,语法格式为 var 类名;//此时对象名还只是个指针,没分配到内存。

然后调用create 分配内存 语法格式为  对象名:=类名.create。

对象中数据域(C++中的数据成员)和方法的引用 语法如下:

  对象.数据域;

  对象.方法;

  释放对象 对象.free;

3.类的封装

类的每个成员都有一个称为可见性的属性,我们用下面的关键字之一来表示它:private、protected、 public、published 和 automated

  类的的数据域(数据成员):

1)Fields  (字段)

字段就像属于对象的一个变量,它可以是任何类型,包括类类型 (也就是说,字段可以存储对象的引用)。 字段通常具有private 属性。

2)方法:

(1)方法的类型

对象的方法能定义成静态( static )、虚拟( virtual )、动态( dynamic)或消息处理(message )。

     1. 静态方法

   静态方法是方法的缺省类型,对它就像对通常的过程和函数那样调用。编译器知道这些方法的地址,所以调用一个静态方法时它能把运行信息静态地链接进可执行文件。 静态方法执行的速度最快,但它们却不能被覆盖来支持多态性。

     2. 虚拟方法(v i r t u a l)

虚拟方法和静态方法的调用方式相同。由于虚拟方法能被覆盖,在代码中调用一个指定的虚拟方法时编译器并不知道它的地址。因此,编译器通过建立虚拟方法表 ( V M T )来查找在运行时的函数地址。所有的虚拟方法在运行时通过V M T来调度,一个对象的V M T表中除了自己定义的虚拟方法外,还有它的祖先的所有的虚拟方法,因此虚拟方法比动态方法用的内存要多,但它执行得比较快。(速度优化)

{虚拟方法还有一种特例,即抽象方法:例如

    procedure One;override;abstract;

在One方法后面,不但有override关键字,还多了一个abstract关键字(意为抽象)。这种方法称为抽象方法(在C++中称为纯虚拟函数)。含有抽象方法的类称为抽象类。抽象方法的独特之处在于,它只有声明,而根本没有实现部分,如果你企图调用一个对象的抽象方法,你将得到一个异常。只有当这个类的派生类重载并实现了该方法之后,它才能够被调用。(在C++中,甚至根本就不能建立一个抽象类的实例。)

既然如此,那么这种抽象方法又有什么用呢?

抽象方法本身不能够做任何事情,必须在子类中被重载并实现,才能够完成有意义的工作。但抽象方法的存在,相当于为父类留下了一个接口,当程序将一个子类的对象赋予父类的变量时,父类的变量就可以调用这个方法,当然此时它运行的是相应的子类中重载该方法的代码。如果没有这个抽象方法,父类的变量就不能调用它,因为它不能调用一个只在子类中存在、而在父类中不存在的方法!

     3. 动态方法 (d y n a m i c)

动态方法跟虚拟方法基本相似,只是它们的调度系统不同。编译器为每一个动态方法指定一个独一无二的数字,用这个数字和动态方法的地址构造一个动态方法表 ( D M T )。不像V M T表,在D M T表中仅有它声明的动态方法,并且这个方法需要祖先的D M T表来访问它其余的动态方法。正因为这样,动态方法比虚拟方法用的内存要少,但执行起来较慢,因为有可能要到祖先对象的D M T中查找动态方法。(代码大小优化)

     4. 消息处理方法 (message)

     在关键字m e s s a g e后面的值指明了这个方法要响应的消息。用消息处理方法来响应windows的消息,这样就不用直接来调用它。。声明方法时,通过包含message指示字创建,并在message后面跟一个介于+1~49151之间的整数常量,体指定消息的号码(ID)。一个message方法必须具有一个单一var参数的过程。例

Typt

Ttextbox=class(Tcustomcontrol)

Private WMChar(var Message:TWMChar);message WM_CHAR;

Endl;

 

(2)方法的覆盖

     在Object Pascal覆盖一个方法用来实现O O P的多态性概念。通过覆盖使一方法在不同的派生类间表现出不同的行为。Object Pascal 中能被覆盖的方法是在声明时被标识为v i r t u a l或d y n a m i c的方法。为 了覆盖一个方法,在派生类的声明中用o v e r r i d e代替v i r t u a l或d y n a m i c 。如果不含override关键字,如果子类声明和父类相同的方法,则视为新建。

T1=class

Procedure Draw;virtual;

End;

T2=calss(T1)

Procedure Draw;override;

End;

(3)方法的重载

所有方法都支持重载。重载后,子类可以含有和祖先类的方法具有不同参数的方法,它只是重载了该方法,并没覆盖它,调用时编译器依靠参数决定到底调用哪一个。能重载的方法必须用o v e r l o a d指示符标识出来。若要重载的是虚方法则需要使用reintroduce指示字。

如 (重载虚方法)

type

  T1=class(Tobject)

     Procedure test(I:integer);virtual;

End;

T2=calss(T1)

Procedure Test(S:string);reintroduce;oveload;;

End;

(4)方法的继承

例如,你如果使用以下代码:

  type

  TMyClass = class

  procedure One;virtual;

  end;

  

  type

  TNewClass = class(TMyClass)

  procedure One;override;

  end;

  

  procedure TMyclass.One;virtual;

  begin

  ShowMessage(´调用了TMyclass的方法!´);

  end;

  

  procedure TNewClass.One; override;

  begin

  Inherited;

  ShowMessage(´调用了TNewClass的方法!´);

  end;

  

  可以看到,从TMyClass派生了一个新类TNewClass。这两个类都声明了一个名字相同的方法One。所不同的是,在TMyClass中,One方法后面多了一个Virtual关键 字,表示这个方法是一个虚拟方法(Virtual Method)。而在TNewClass中,One方 法后面多了一个Override关键字,表示该方法进行了重载(Override)。重载技术能够实现许多特殊的功能。

让我们来仔细分析它们的实现部分。在TMyclass.One方法的实现部分,调用ShowMessage过程弹出一个对话框,说明该方法已被调用;这里没有任何特别的地方。在TNewClass.One方法中,出现了一条以前从未出现过的语句:

Inherited;

这个词的中文意思是“继承”。我们暂时不要去涉及到太过复杂的OOP概念,只要知道这条语句的功能就是了。它的功能是调用基类中相同的虚拟方法中的代码。

 那么程序将弹出两次对话框,第一次是调用TMyclass类中的One方法,第二次才是TNewClass.One方法中的代码。

重载技术使得我们不但可以在派生类中添加基类没有的数据和方法,而且可以非常方便地继承基类中原有方法的代码,只需要简单地加入Inherited就可以了。如果你不加入Inherited语句,那么基类的相应方法将被新的方法覆盖掉。

(4)方法的构造函数和析构函数

  Constructors Create;

Destructor Destroy;

使用 对象,Create;

          对象.Destroy;(最好使用 对象.free)

3)属性

普通属性

我们在delphi的类中常常能看到这样的代码:

propert property 属性名 类型名 read 字符串1 write 字符串2

例如:

property Left: Integer read FLeft write SetLeft;

我们在private中看到申明:

procedure SetLeft(Value: Integer);(方法)

和如下代码实现:

procedure TControl.SetLeft(Value: Integer);

begin

  SetBounds(Value, FTop, FWidth, FHeight);

  Include(FScalingFlags, sfLeft);

end;

如果你写了如下代码改变left:control1.left:=23,那么程序调用了函数SetLeft(23),SetBounds是改变区域的函数,这里你就明白了它封装了的好处,每次你改变left时它就会根据新的left而改变区域的大小,这个函数同时也改变了Fleft的大小。

这样外部就看起来只是通过赋值运算来改变了该属性的值。Read和write可以是变量,或者是函数,取决于你的设计。

你当然可以这样写:

propert property 属性名 类型名 read 变量1(函数) write 变量2(函数)。

变量1和变量2可以是相同的。你也可以这样

propert property 属性名 类型名 read 方法1 write 方法2。

任你组合。但是有2点要注意:

1.命名规则最好按习惯来,易于阅读。

2. 如果是变量,那么类型要和属性的类型一致,如果是方法,那么入口参数要和属性的类型一致。

声明部分:

 Type

Tdog=calss

Private

Color:string;

Function Getcolor(x:string);

Procedure Setetcolor(x:string)

Public

Property x:string read Getcolor write setcoler;

End;

实现部分:

Fenction Tdog.Getcolor :string;

Begin

Result:=color;

End;

Procedure Tdog.setcoler(value:string)

Begin

Color:=value;

End;

事件属性Tevent

我们常常使用组件的事件属性,比方说click事件,可是我们很难从表面看出它是如何调用的呢,如何触发的呢。下面我来给你解答。

我们在属性管理器object inspector中看到event页onclick右边对应了一个方法的名字。我们其实可以这样给一个组件的事件对应上一个出来方法。以一个form为例子Form1. OnMouseDown:=‘你的方法‘。注意方法的入口参数有讲究,这里是(Sender:TObject)

我们还是一tcontrol为例子,我们找到这段代码:

property OnMouseDown: TMouseEvent read FOnMouseDown write FOnMouseDown;跟上面讲的类似,不过这里有个特殊的类型,TNOtifyEvent,是个事件类型,我们找到它的申明:

TMouseEvent = procedure(Sender: TObject; Button: TMouseButton;Shift: TShiftState; X, Y: Integer) of object;

可以看到,它其实就是个函数,但是蓝色部分把入口参数限定了。那么我们通过赋值Form1. OnMouseDown:=‘你的方法‘,就对应了OnMouseDown的方法。然后我们只要写了一段拦截鼠标消息的函数,在里面直接或间接调用FonMouseDown,那么就把消息和处理函数对应上去了。这里它间接调用的层数比较多,讲起来比较费时间,涉及到Message类型,建议大家去看下李维的书。

以下附上间接调用过程,其实还要很多消息发生时也间接调用了,就不一一举出来了:(

 

procedure WMRButtonDblClk(var Message: TWMRButtonDblClk); message WM_RBUTTONDBLCLK;//拦截消息的函数

procedure TControl.WMRButtonDblClk(var Message: TWMRButtonDblClk);

begin

  inherited;

  DoMouseDown(Message, mbRight, [ssDouble]);

end;

 

procedure DoMouseDown(var Message: TWMMouse; Button: TMouseButton;

Shift: TShiftState);

procedure TControl.DoMouseDown(var Message: TWMMouse; Button: TMouseButton;

  Shift: TShiftState);

begin

  if not (csNoStdEvents in ControlStyle) then

    with Message do

      if (Width > 32768) or (Height > 32768) then

        with CalcCursorPos do

          MouseDown(Button, KeysToShiftState(Keys) + Shift, X, Y)

      else

        MouseDown(Button, KeysToShiftState(Keys) + Shift, Message.XPos, Message.YPos);

end;

 procedure MouseDown(Button: TMouseButton; Shift: TShiftState;

X, Y: Integer); dynamic;

procedure TControl.MouseDown(Button: TMouseButton;

  Shift: TShiftState; X, Y: Integer);

begin

  if Assigned(FOnMouseDown) then FOnMouseDown(Self, Button, Shift, X, Y);

end;

 

好处:

如果你多写自己的类,你会发现这样做是多么的方便, delphi你都只是调用contol1.text来访问,control1.text:=’某字符串’来修改它的值。

而在处理消息方面,基类把onclick,onmousedown这样的属性申明为protected,如果你要使用,可以申明为published就可以出现在object inspector里面,然后方便的写处理方法,你也可以不公开,而在ctreate函数中给它赋值,而不用像java那样,写listener那么复杂。

5.类的多态

 虚方法和动态方法通过覆盖和重载技术实现多态。

6.类的继承

7.类方法

类方法是作用在类而不是对象上面的方法(不同于构造函数)。类方法的定义必须以关键字class 开始,

比如,

  type

     TFigure = class

     public

       class function Supports(Operation: string): Boolean; virtual;

       class procedure GetInfo(var Info: TFigureInfo); virtual;

 

       ...

 

     end;

类方法的定义部分也必须以 class 开始,比如,

  class procedure TFigure.GetInfo(var Info: TFigureInfo);

  begin

 

     ...

 

  end;

在类方法的定义部分,Self 表示调用方法的类

类方法既可以通过类引用来调用,也可以使用对象,当使用后者时, Self 值等于对象所属的类。

7.Self参数:

Self是指所编的程序范围是在哪一个类中,Delphi中大都在窗体范围内编程,因此,Self即指窗体,假如在编写一个类或是一个组件,则Self指该类或该组件.我们在过程和函数的声明中可以看出Self是代表哪个组件,即Self代表"."号之前的组件.另外应注重,Self只能用在类方法中,而不能用在过程或函数中.

delphi 基础之二 面向对象概念初步

标签:

原文地址:http://www.cnblogs.com/bjxsky/p/4610249.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!