posts - 12,  comments - 8,  trackbacks - 0
http://www.cnblogs.com/keyvip/archive/2010/11/06/1870605.html

作者:巴哈姆特
(转载请注明出处并保持完整)

这回,我们讨论怎么编写我们自己的第一个类。
    在编写我们自己的类之前,首先要说的是“类的继承”。
    记得前几天,我在和一个朋友讨论类的特点的时候,他说:“类是可以没有构造方法的!”其实类必须有至少一个构造方法的,但是他的话也不全错,可以理解成“我们可以不实现我们自己的构造方法”。
    当我们没有显式的为类编写一个构造方法的时候,那么,看上去,这个类好象是没有构造方法,但是实际上,就算你没有为这个类编写一个属于你自己的构造方法的时候,该类还是有构造方法的;为什么呢?因为,在你选定父类并从父类派生(继承)一个新的后代类的时候,首先,编译器会COPY一份父类中“可继承”的数据成员到新的子类中。
    那么,什么是可继承的数据成员呢,我们来看下面的代码:

TPersistent = class(TObject)
private
  procedure AssignError(Source: TPersistent);
protected
  procedure AssignTo(Dest: TPersistent); virtual;
  procedure DefineProperties(Filer: TFiler); virtual;
  function  GetOwner: TPersistent; dynamic;
public
  destructor Destroy; override;
  procedure Assign(Source: TPersistent); virtual;
  function  GetNamePath: string; dynamic;
end;
 

上面是VCL内部TPersistent类的声明代码,我们可以看到几个关键字private、protectedpublic,这几个关键字是说明数据成员的访问限制的,其中:

private:     存在于该访问限制下的数据成员只能被本类所访问。
protected:   存在于该访问限制下的数据成员只能被本类及后代类所访问。
public:      存在于该访问限制下的数据成员可以被任何类访问。
published:   存在于该访问限制下的数据成员的访问限制与public一样,但是在本域下的属性将会在注册类的时候被注册给IDE。
automated:   存在于该访问限制下的数据成员为自动成员,常用于定义嵌入(OLE)自动化类型信息的公共接口。 PS: 该域并不常用,这里不做介绍
 

    由此可见,所谓的“可继承”部分是指除"private"域指定的数据成员。
    回到上面的问题,由于在继承新类的时候,编译器会COPY一份父类所有可继承部分的数据成员到新的子类中,当没有为新类编写构造方法时,新类会调用父类的构造方法,如果父类中也没有编写构造方法,那么将层层上溯,直到根类TObject中。
    就像上面的类,我们可以看到,在类里面的确是没有重写新的构造方法,但是这个类的构造方法却是真实的存在的。

    现在,我们可以开始编写第一个属于我们自己的简单的类了:

type // 说明我们从这里开始往下要声明的是自定义的数据类型
  TNewClass = class(TPersistent)
    // TNewClass   新类的类名
    // TPersistent 父类的类名
  private
    // 私有域数据成员
  protected
    // 保护域数据成员
  public
    // 公共域数据成员
    procedure SayHello(); // 我们为类添加的新的方法
  published
  end; // 类的声名结束
 
implementation // 实现部分
 
procedure TNewClass.SayHello;
  // SayHello 方法的实现部分
begin
  ShowMessage('Hello');
end;
 

至此,我们完成了一个简单的类的编写。
    我们可以写一段简单的代码来测试类的工作情况:

var
  NewClass: TNewClass; // 定义一个类 TNewClass 的对象 NewClass
begin
  NewClass:= TNewClass.Create; // 为对象 NewClass 分配内存空间,并创建它
  NewClass.SayHello(); // 调用 SayHello 方法
  NewClass.Free; // 销毁对象并释放对象的内存空间
end;
 

有时,我们会看到这样的声明方式:

type
  TNewClass = class
    ...
  end;
 

这是一种省略父类的声明方式,当省略了父类的时候,DELPHI编译器会把根类TObject当做新类的基类,和

type
  TNewClass = class(TObject)
    ...
  end;
 

效果一样,这里我推荐下面的方式,因为这样可以让代码更清晰。

 

posted on 2014-09-26 10:59 hcwp 阅读(...) 评论(...) 编辑 收藏