访问祖先类的虚方法(直接访问祖先类的VMT,但是这种方法在新版本中未必可靠)
访问祖先类的虚方法
问题提出
在子类覆盖的虚方法中,可以用inherited调用父类的实现,但有时候我们并不需要父类的实现,而是想跃过父类直接调用祖先类的方法。
举个例子,假设有三个类,实现如下:
type
TClassA = class
procedure Proc; virtual;
end;
TClassB = class(TClassA)
procedure Proc; override;
end;
TClassC = class(TClassB)
procedure Proc; override;
end;
implementation
procedure TClassA.Proc;
begin
ShowMessage('Proc of class A');
end;
procedure TClassB.Proc;
begin
ShowMessage('Proc of class B');
end;
procedure TClassC.Proc;
begin
ShowMessage('Proc of class C');
end;
用如下代码调用虚方法Proc:
var
C: TClassA;
begin
C := TClassC.Create;
C.Proc;
C.Free;
end;
我们知道最终调用的是TClassC.Proc;如果在TClassC.Proc中加上Inherited,则TClassB.Proc可以得到调用;但是现在,若想在TClassC.Proc中直接调用TClassA.Proc,该怎么办呢?
解决之道
如果是C++,只需要这样写:TClassC::Proc。
在Delphi却没有办法做到,Delphi不允许我们跃级调用祖先类的方法。尽管如此,还是能从另一个角度来寻求解决的办法。
解决之道就是VMT,每一个类就是一个指向VMT的指针,而VMT的作用其实就是用来保存虚方法的。在VMT的正方向上,列着从祖先类起的所有虚方法,只需要偏移TClassA的VMT到Proc,然后调用之即可。
来看看这个问题是怎么得解决的:
procedure TClassC.Proc;
type
TProc = procedure of object;
var
M: TMethod;
begin
M.Code := PPointer(TClassA)^;
M.Data := Self;
TProc(M)();
ShowMessage('Proc of class C');
end;
执行一次调用,可以看到先弹出:Proc of class A;然后弹出:Proc of class C。这说明TClassA.Proc在TClassC.Proc中被调用到了。
请注意上面的代码,TClassA的VMT上的第0偏移就是Proc的地址,而TClassA继承自TObject,TObject本身也有一些虚方法的,比如AfterConstruction,那么这些是存放在哪里呢?
秘密就在VMT的负偏移上,在System单元中声明了虚表的结构偏移,在负方向上有AfterConstruction的进入点。需要指出的是,System单元中声明了结构偏移正方向的几个已经过时了,第0偏移(vmtQueryInterface)不是存放QueryInterface,而是存放第一个虚方法(除TObject外)。
下面是从帮助上拷下来的VMT布局:
Offset Type Description
-76 Pointer pointer to virtual method table (or nil)
-72 Pointer pointer to interface table (or nil)
-68 Pointer pointer to Automation information table (or nil)
-64 Pointer pointer to instance initialization table (or nil)
-60 Pointer pointer to type information table (or nil)
-56 Pointer pointer to field definition table (or nil)
-52 Pointer pointer to method definition table (or nil)
-48 Pointer pointer to dynamic method table (or nil)
-44 Pointer pointer to short string containing class name
-40 Cardinal instance size in bytes
-36 Pointer pointer to a pointer to ancestor class (or nil)
-32 Pointer pointer to entry point of SafecallException method (or nil)
-28 Pointer entry point of AfterConstruction method
-24 Pointer entry point of BeforeDestruction method
-20 Pointer entry point of Dispatch method
-16 Pointer entry point of DefaultHandler method
-12 Pointer entry point of NewInstance method
-8 Pointer entry point of FreeInstance method
-4 Pointer entry point of Destroy destructor
0 Pointer entry point of first user-defined virtual method
4 Pointer entry point of second user-defined virtual method
后记
利用虚表调用虚方法的做法,终究不是安全的,因为Borland(CodeGear)没有向你保证每一个Delphi版本的VMT布局都是一样的。
因此,使用这个方法的时候要慎之又慎。
http://blog.csdn.net/linzhengqun/article/details/1755493
-------------------------------------------------------------------------------
我将这个例子改造,变成2个虚函数:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, ExtCtrls, StdCtrls;
type
TForm1 = class(TForm)
Image1: TImage;
Button1: TButton;
Button2: TButton;
procedure Button1Click(Sender: TObject);
procedure Button2Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end; type
TClassA = class
procedure Proc; virtual;
procedure second; virtual;
end; TClassB = class(TClassA)
procedure Proc; override;
procedure second; override;
end; TClassC = class(TClassB)
procedure Proc; override;
procedure second; override;
end; var
Form1: TForm1; implementation
{$R *.dfm} procedure TClassA.Proc;
begin
ShowMessage('Proc of class A');
end;
procedure TClassB.Proc;
begin
ShowMessage('Proc of class B');
end; procedure TClassA.second;
begin
ShowMessage('second of class A');
end;
procedure TClassB.second;
begin
ShowMessage('second of class B');
end; procedure TClassC.Proc;
type
TProc = procedure of object;
var
M: TMethod;
P: Pointer;
begin
P := PPointer(TClassA)^;
M.Code := p;
M.Data := Self;
TProc(M)();
ShowMessage('Proc of class C');
end; procedure TClassC.second;
type
TProc = procedure of object;
var
M: TMethod;
p: Pointer;
begin
// P := PPointer(TClassA)^;
// P := Pointer(Integer(p)+4); // 错误:这里试图取得VMT的第二个函数
p:= PPointer(integer(TClassA) + )^;
M.Code := P;
M.Data := Self;
TProc(M)();
ShowMessage('second of class C');
end; procedure TForm1.Button1Click(Sender: TObject);
var
C: TClassA;
begin
C := TClassC.Create;
C.Proc;
C.Free;
end; procedure TForm1.Button2Click(Sender: TObject);
var
C: TClassA;
begin
C := TClassC.Create;
C.Second;
C.Free;
end; end.
// 另外改成P := PPointer(TClassC)^; 也不行,这是为什么?
[石家庄]王烨 2016/3/21 14:36:41
第二个声明一个 procedure of object,然后又去用一个method强制转换
第一个方法的地址
不是VMT
也就是VMT的第一个元素的值
而且他这种方法
只能取virtual的
不能取dynamic的
而且他这种,如果带参数咋办?
用我那种吧
这样不需要按照Method方式调用
访问祖先类的虚方法(直接访问祖先类的VMT,但是这种方法在新版本中未必可靠)的更多相关文章
- 【译文】 C#面向对象的基本概念 (Basic C# OOP Concept) 第一部分(类,对象,变量,方法,访问修饰符)
译文出处:http://www.codeproject.com/Articles/838365/Basic-Csharp-OOP-Concept 相关文档:http://files.cnblogs.c ...
- Java 访问限制符 在同一包中或在不同包中:使用类创建对象的权限 & 对象访问成员变量与方法的权限 & 继承的权限 & 深入理解protected权限
一.实例成员与类成员 1. 当类的字节码被加载到内存, 类中类变量.类方法即被分配了相应内存空间.入口地址(所有对象共享). 2. 当该类创建对象后,类中实例变量被分配内存(不同对象的实例变量互不相同 ...
- java中,方法可以访问他的类对象的任何私有特性
java中,方法可以访问他的类对象的任何私有特性 读一本书(Core Java for the Impatient)时,发现这个注意,以前的时候没有在意,今天仔细想想发现记忆不深刻.记录一下 下面代码 ...
- C#类、方法的访问修饰符
这篇文章主要介绍了C#类的访问修饰符用法,较为详细的分析了C#类的访问修饰符概念与用法,具有一定的参考借鉴价值,需要的朋友可以参考下 本文详细分析了C#类的访问修饰符用法,分享给大家供大家参考.具体用 ...
- accessor mothod mutator mothod 更改器方法 访问器方法 类的方法可以访问类的任何一个对象的私有域!
LocalDate.plusDate String.toUpperCase GregorianCalendar.add import java.time.*; public class Calenda ...
- LindAgile~缓存拦截器支持类的虚方法了
写它的原因 之前写过一个缓存拦截器,主要在方法上添加CachingAspect特性之后,它的返回值就可以被缓存下来,下次访问时直接从缓存中返回结果,而它有一个前提,就是你的方法需要是一个接口方法,缓存 ...
- 《Effective Java》笔记 使类和成员的可访问性最小化
类和接口 第13条 使类和成员的可访问性最小化 1.设计良好的模块会隐藏所有的实现细节,把它的API与实现清晰的隔离开来,模块之间只通过它们的API进行通信,一个模块不需要知道其他模块的内部工作情况: ...
- [C++]虚函数-同名访问
首先来看一下派生类和基类成员同名事的处理规则: 派生类内定义了一个与基类同名的成员,该现象称为同名覆盖,此时,无论派生类内部成员函数还是派生类的对象访问同名成员,如果未加任何特殊标识,则访问派生类中重 ...
- Effective Java:Ch4_Class:Item13_最小化类及其成员的可访问性
要区别一个模块是否设计良好,最重要的因素是,对于其他模块而言该模块隐藏其内部数据和其他实现细节的程度.设计良好的模块应该隐藏所有实现细节,将API与其实现清晰地隔离开来.这样,模块之间通过他们的API ...
随机推荐
- 【转】如何在CentOS/RHEL中安装基于Web的监控系统 linux-das
Linux-dash是一款为Linux设计的基于Web的轻量级监控面板.这个程序会实时显示各种不同的系统属性,比如CPU负载.RAM使用率.磁盘使用率.网速.网络连接.RX/TX带宽.登录用户.运行的 ...
- 自己用js写的两个日历控件
前一阵写了两个日历控件,做了简单的封装,发出来共朋友们参考. 第一个日历控件,条状的日历. (使用方法:调用initBarTime(id,evn),第一个参数是要渲染div的id,第二个参数是点击日期 ...
- nm命令
它用来列出一个目标文件中的各种符号. nm命令还是比较简单而且强大的.它用来列出一个目标文件中的各种符号.符号的种类很多,以下是一些常见的符号类型 nm输出字符 含义 R Read only symb ...
- [置顶] c#验证码识别、图片二值化、分割、分类、识别
c# 验证码的识别主要分为预处理.分割.识别三个步骤 首先我从网站上下载验证码 处理结果如下: 1.图片预处理,即二值化图片 *就是将图像上的像素点的灰度值设置为0或255. 原理如下: 代码如下: ...
- 程序启动报错:ORA-12505;PL/SQL却可以登录的解决方法
一.异常{ ORA-12505, TNS:listener does not currently know of SID given in connect descriptor The Connect ...
- encode_utf8 把字符编码成字节 decode_utf8解码UTF-8到字符
encode_utf8 $octets = encode_utf8($string); Equivalent to "$octets = encode("utf8", $ ...
- 基于visual Studio2013解决面试题之1503最大公约数最小公倍数
题目
- 阿根廷探戈(Argentine Tango)舞步
阿根廷探戈(Argentine Tango)舞步 阿根廷探戈(Argentine Tango)舞步 2011-11-22 13:05:11 不像其它大部分的社交舞,阿根廷探戈没有固定的舞步,它是一 ...
- 621 - Secret Research
Secret Research At a certain laboratory results of secret research are thoroughly encrypted. A res ...
- 基于visual Studio2013解决面试题之0410计算二进制中1的个数
题目