我需要澄清Delphi Type Casting
我写了两个类的例子:TClassA和TClassB,TClassB派生自TClassA。
代码如下:
program TEST;
{$APPTYPE CONSOLE}
uses
System.SysUtils;
type
TClassA = class(TObject)
public
Member1:Integer;
constructor Create();
function ToString():String; override;
end;
type
TClassB = class(TClassA)
public
Member2:Integer;
constructor Create();
function ToString():String; override;
function MyToString():String;
end;
{ TClassA }
constructor TClassA.Create;
begin
Member1 := 0;
end;
function TClassA.ToString: String;
begin
Result := IntToStr(Member1);
end;
{ TClassB }
constructor TClassB.Create;
begin
Member1 := 0;
Member2 := 10;
end;
function TClassB.MyToString: String;
begin
Result := Format('My Values is: %u AND %u',[Member1,Member2]);
end;
function TClassB.ToString: String;
begin
Result := IntToStr(Member1) + ' - ' + IntToStr(Member2);
end;
procedure ShowInstances();
var
a: TClassA;
b: TClassB;
begin
a := TClassA.Create;
b := TClassB(a); // Casting (B and A point to the same Memory Address)
b.Member1 := 5;
b.Member2 := 150; // why no error? (1)
Writeln(Format('ToString: a = %s, a = %s',[a.ToString,b.ToString])); // (2)
Writeln(Format('Class Name: a=%s, b=%s',[a.ClassName,b.ClassName])); // (3)
Writeln(Format('Address: a=%p, b=%p',[@a,@b])); // (4)
Writeln(b.MyToString); // why no error? (5)
readln;
end;
begin
try
ShowInstances;
except
on E: Exception do
Writeln(E.ClassName, ': ', E.Message);
end;
end.
程序输出为:
ToString: a = 5, a = 5
Class Name: a=TClassA, b=TClassA
Address: a=0012FF44, b=0012FF40
My Values is: 5 AND 150
(1)会员2的地址是什么?这可能是“访问违规”吗?
(2)ok,ToString()方法指向同一地址
(3)为什么a和b具有相同的ClassName?
(4)ok,a和b是两个不同的变量
(5)如果b是TClassA,为什么可以使用“MyToString”方法?
答案 0 :(得分:23)
您正在对变量应用 hard 类型转换。当你这样做时,你告诉编译器你知道你在做什么,编译器信任你。
(1)会员2的地址是什么?这可能是“访问冲突”吗?
当您为类的成员分配值时,编译器会使用该变量的类定义来计算该成员在内存空间中的偏移量,因此当您有类似这样的类声明时:
type
TMyClass = class(TObject)
Member1: Integer; //4 bytes
Member2: Integer; //4 bytes
end;
此对象的内存中表示如下所示:
reference (Pointer) to the object
|
|
--------> [VMT][Member1][Member 2][Monitor]
Offset 0 4 8 12
当您发出如下声明时:
MyObject.Member2 := 20;
编译器只使用该信息来计算应用该赋值的内存地址。在这种情况下,编译器的赋值可以转换为
PInteger(NativeUInt(MyObject) + 8)^ := 20;
所以,你的任务成功只是因为(默认)内存管理器的工作方式。当您尝试访问不属于您的程序的内存地址时,操作系统会发起AV。在这种情况下,您的程序从操作系统获取的内存超过了所需的内存。恕我直言,当你没有获得AV时,你实际上是不吉利的,因为你的程序内存现在可能会被无声地破坏。碰巧驻留在该地址的任何其他变量可能已更改其值(或元数据),并且会导致未定义的行为。
(2)ToString()方法指向相同的地址
由于ToString()方法是虚方法,因此该方法的地址存储在VMT中,并且在运行时确定调用。请查看What data does a TObject contain?,并阅读参考书籍章节:The Delphi Object Model。
(3)为什么a和b具有相同的ClassName?
类名也是对象的运行时元数据的一部分。您将错误的模具应用于对象这一事实并不会改变对象本身。
(4)a和b是两个不同的变量
当然,你宣布了它,看看你的代码:
var
a: TClassA;
b: TClassB;
嗯,两个不同的变量。在Delphi中,对象变量是引用,因此,在某些代码行之后都引用相同的地址,但这是另一回事。
(5)如果b是TClassA,为什么可以使用“MyToString”方法?
因为你告诉编译器没问题,如上所述,编译器信任你。这很糟糕,但Delphi也是一种低级语言,如果你愿意,你可以做很多疯狂的事情,但是:
如果你想(并且你肯定希望大部分时间)安全,不要在你的代码中应用这样的硬强制转换。使用as operator:
as 运算符执行检查的类型转换。表达式
对象 as class
返回对象作为同一对象的引用,但是使用类给出的类型。在运行时,对象必须是由类或其后代之一表示的类的实例,或者为nil;否则会引发异常。如果声明的对象类型与类无关 - 也就是说,如果类型是不同的并且一个不是另一个的祖先 - 则会产生编译错误。
因此,使用as运算符,无论是在编译时还是在运行时,都是安全的。
将您的代码更改为:
procedure ShowInstance(A: TClassA);
var
b: TClassB;
begin
b := A as TClassB; //runtime exception, the rest of the compiled code
//won't be executed if a is not TClassB
b.Member1 := 5;
b.Member2 := 150;
Writeln(Format('ToString: a = %s, a = %s',[a.ToString,b.ToString]));
Writeln(Format('Class Name: a=%s, b=%s',[a.ClassName,b.ClassName]));
Writeln(Format('Address: a=%p, b=%p',[@a,@b]));
Writeln(b.MyToString);
readln;
end;
procedure ShowInstances();
begin
ShowInstance(TClassB.Create); //success
ShowInstance(TClassA.Create); //runtime failure, no memory corrupted.
end;
答案 1 :(得分:4)
Member2
的地址未由内存管理器分配。写入Member2
的可能结果是堆损坏,后续访问冲突在程序的完全不同的部分。这是一个非常讨厌的bug,编译器无法帮助你。在进行不安全的类型转换时,您必须知道自己在做什么。
这是因为ToString
方法是虚方法,因此其地址由正在创建的类实例的实际类型决定。如果您通过静态替换虚方法(在您的情况下通过override
替换reintroduce
指令),结果将会有所不同。
因为ClassName
方法也是虚拟的(实际上不是VMT的成员,但这是无关紧要的实现细节)。
是的,a
和b
是对同一个实例的两次引用。
因为ToMyString
方法是静态的。实例的实际类型对静态方法无关紧要。
答案 2 :(得分:3)
(1)会员2的地址是什么?这可能是“访问冲突”吗?
是。 AV是可能的。在你的情况下,你有运气:)
好的,ToString()方法指向同一个地址
是的,因为VTable在创建时担心。
(3)为什么a和b具有相同的ClassName?
与(2)中的答案相同。
(4)好的,a和b是两个不同的变量
不是真的。你从堆栈打印地址:)
(5)如果b是TClassA,为什么可以使用“MyToString”方法?
b是TClassB,但错误指向TClassA实例。
您应该使用作为运算符进行此类强制转换。在这种情况下,它会失败。