gpt4 book ai didi

delphi - 静态类方法和常规例程指针的兼容性如何?

转载 作者:行者123 更新时间:2023-12-03 15:47:10 24 4
gpt4 key购买 nike

在我看来,从实际的角度来看,静态类方法和常规例程指针是兼容的,但编译器不知道这一点。示例:

type
TFunc = function(i: Integer): string;

TMyClass = class
public
class function StaticMethod(i: Integer): string; static;
end;

class function TMyClass.StaticMethod(i: Integer): string;
begin
Result := '>' + IntToStr(i) + '<';
end;

function GlobalFunc(i: Integer): string;
begin
Result := '{' + IntToStr(i) + '}';
end;

procedure CallIt(func: TFunc);
begin
Writeln(func(42));
end;

begin
CallIt(TMyClass.StaticMethod); // 1a: doesn't compile
CallIt(GlobalFunc); // 1b: compiles

CallIt(@TMyClass.StaticMethod); // 2a: compiles iff $TYPEDADDRESS OFF
CallIt(@GlobalFunc); // 2b: compiles iff $TYPEDADDRESS OFF

CallIt(Addr(TMyClass.StaticMethod)); // 3a: compiles
CallIt(Addr(GlobalFunc)); // 3b: compiles

Readln;
end.

正如评论中所指出的,3a 和 3b 都可以编译(其中编译包括在这个简单示例中在运行时工作)。当且仅当 $TYPEDADDRESSOFF 时,2a 和 2b 才会编译。但 1a/1b 是不同的:1b 总是编译,而 1a 从不编译。这是设计上的区别吗?使用 3a 保存还是我忽略了任何陷阱?

最佳答案

具有相同参数和结果类型的静态类函数和普通函数在二进制级别上没有区别 - 它们是二进制兼容的,所以你的例子没问题。当然,对于编译器来说它们是不同的类型,因此您需要 Addr()@ 来编译您的示例。

Addr() 等同于 @ 运算符,只不过它不受 $T 编译器指令的影响。如果您在示例上切换类型检查,则不会编译:

{$T+}
begin
CallIt(@TMyClass.StaticMethod);
Readln;
end.

[Pascal 错误] Project10.dpr(28): E2010 不兼容类型:“TFunc”和“指针”

关于delphi - 静态类方法和常规例程指针的兼容性如何?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6940171/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com