delphi之self
在使用delphi的对象技术的时候,经常会看到一个词汇:self,它到底指的是什么呢?
我们还要从对象与类的关系谈起。 类是对将要创建的对象的性质的描述,是一种文档。这很重要:类只是一段描述性的文字,它并不会真去分配内存,无论在其中定义多少变量。 如果打个比方,类就是图纸,而对象就是根据图纸盖的房子。对象是真正在 内存中存在的东西,是运行“实体”。根据一份图纸可以盖多个相似的房子,同 样道理,根据一个类,可以创建多个类似的对象,这个过程叫做“实例化”。在delphi中使用对象技术,要遵循以下的步骤:
1。定义一个类
2。用该类声明一个名字(实质是一个指针)
3。用该类实例化一个对象,并使它与先前的名字联系起来
4。调用对象的方法或属性
5。释放对象
下面我们写一个最简单的表达累加器功能的类
type
TCount = class
private
FNum: integer; // 记录有多少个数字被累加
FSum: integer; // 当前的累加和
pubic
procedure Add(n: integer) // 把整数n累加进去
procedure clear; // 清零
procedure show; // 显示当前信息
constructor create; // 构造函数
end;
....
使用的过程是这样的:
var a: TCount; // 这里只是声明了一个名字,并非真正地分配了一个对象
...... // a这个变量只占用4字节地内存
a := TCount.create; // 在堆空间中分配内存,并把首地址拷贝到a中。
a.add(5);
a.add(7);
a.show(); // 以上是调用a对象的有关方法
a.free; // free会去调用析构函数,完成堆空间的释放。
我们看这样一个问题:一个对象到底占用多大的内存空间?
答案是很小!因为在分配一个对象地时候,实际上只分配了类中定义的数据,而没有分配其中的函数所需要的空间。这些类中定义的函数(称为成员函数),与普通函数一样被存放在静态地址空间中。
这样就引出了第二个问题:既然对象的数据和操作这些数据的方法不是存在一起,那么这些函数如何才能知道到底要操作哪个数据块呢?(类可生成多个实例)显然,最容易想到的解决办法就是让这些特殊函数带一个参数,是个指针类型,该指针指示要操作的数据块的首地址。
事实上,成员函数正是这样做的,它们有一个缺省的参数self,这是个指针类型,对于上边的例子,它的定义就是:TCount self。编译器在遇到调用a.add(5)的时候,把它解释为:TCount.add(a,5);把代表对象的数据块的地址送给add函数作为第一个参数--隐含的参数。说得本质一些就是:self是当前正在执行本函数的那个对象的数据块的首地址。
self既然代表对象自己,那么难道自己还用定义吗?看下边的代码:
procedure TForm1.button1click(sender: TObject);
var
a: TButton;
begin
a := Tbutton.create(self);
....
end;
在创建Tbutton类的对象的时候,需要给出一个内存管理者。
self就表示了正在运行button1click这个过程的那个对象(的数据块的首地址)。其实大多数情况下就是form1。Object pascal参考手册中关于self的讲述:Self(Self 变量)在实现方法时,标志符Self 引用方法所属的对象。比如,下面是Classes 单元中TCollection 的Add 方法的实现:
function TCollection.Add: TCollectionItem;
begin
Result := FItemClass.Create(Self);
end;
Add 方法调用FItemClass 的Create 方法,而FItemClass 所属的类总是TCollectionItem 的派生类,TCollectionItem.Create 有一个TCollection 类型的单一参数,所以,Add 把此时TCollection 的实例传给它,这以下面的代码表示:
var MyCollection: TCollection;
...
MyCollection.Add // MyCollection 被传给TCollectionItem.Create 方法
Self 在很多方面都有用,比如,一个在类中声明的成员(标志符)可能在方法中被重新声明,这种情况
下,你可以使用Self.Identifier 来访问原来的成员。
在类方法的定义部分,Self 表示调用方法的类(which could be a descendant of the class in which it isdefined,它或许是定义方法的类的一个派生类)。若使用类C 调用方法,Self 的类型是class of C(元类)。所以,你不能使用Self 访问字段、属性和平常的方法(由对象调用的方法),但能调用构造函数和其它类方法。
类方法既可以通过类引用来调用,也可以使用对象,当使用后者时, Self 值等于对象所属的类。
http://hi.baidu.com/liuxing167756/blog/item/ec630f1404d35d1bc83d6d8c.html