Delphi 通过名称调用方法[2] - 高级解决方法

Delphi 通过名称调用方法[2]

前面提到了这个常用的方法:Delphi 通过名称调用方法[1]

但这个办法有一个很大的局限性:一旦 TProcedure 声明定了下来,待调用方法的参数表也就一定了。要是我定义了多个待调用方法,且参数个数、类型、返回值均不同,
则这个办法也就无能为力了。另:用 GetProcAddress 代替 MethodAddress 也可以实现类似的效果,不过我们今天讨论的是调用类的“方法”,而它所返回的不是“方法”,
因为 GetProcAddress 仅能取得应用程序的输出(exports)了的过程或函数,这些过程或函数不属于任何类,也就称不上“方法”。当然,效果类似,但是局限也类似 :-(

那么要是参数的个数、类型均可变就无法解决了吗?(要是这样就不会有本文了)通过研究,我发现了一种行之有效的办法:Format 函数(注意,不是 DOS 命令)相传入它的参数的个数和类型不都是可变的吗?只要声明如下:

1
procedure AProc(Param: array of const);

即可这样调用:

1
AProc([123, 'X', True, 'hello'...]);

有朋友可能要说了,那不就简单了,这样不就可以了:

1
2
3
4
5
6
7
8
9
10
11
type
  TProcedure = procedure(Params: array of const) of object;
 
procedure ExecuteRoutine(Obj: TObject; Name: string; Params: array of const);
var
  ...
begin
  ...
    AProcedure(Params);
  ...
end;

别急,问题才刚刚出现呢,你运行试一试?出问题了吧。(为方便起见,暂时称我们的 ExecuteRoutine 函数,为控制函数;待调用方法简称为待调方法)这个形参表的声明
办法的确适合我们的控制函数,但是不适合待调方法。为什么?因为待调方法的形参表的确不是这样(array of const)的啊。当然了,你说你把所有待调方法的形参表都改
成这个样子不就可以了?且不说你需要改动多少东西(包括待调函数的参数表和内部实现,关键是内部实现部分),就看看你改了过后的待调方法的形参表,全部都成了一个
模样。说不定到时候你自己都不知道到底应该传什么参数进去了。因此,我们应该尽量保持待调方法的形参表。
现在问题转化为了在控制函数中已知待调方法的地址及其参数列表(存放在一个 TVarRec 的数组中),如何在调用它的时候将参数传进去。这需要几点预备知识:
  1、首先我们来看看传进来的这个参数表:Params。它的类型被 Delphi 称作可变开放数组(Variant open array),等价于 array of TVarRec,也就是说 Params 是一
个成员为 TVarRec 的数组。换句话说,在参数被传进 Params 的时候,各种类型都被 Delphi 自动转化为了 TVarRec(参见帮助中的 Variant open array parameters 一
节)。看一下 TVarRec 的定义可知,它实际储存的数据域为 4 Bytes,超过 4 Bytes的只存指针,需要注意的是 TVarRec 的大小是 8 Bytes(经研究发现前 4 Bytes 存放
数据,第 5 Byte 为类型)。
  2、调用函数时的参数传递的一般情况(未使用 stdcall 的情况)。对于一般的函数或过程,前三个参数分别放在 EAX、EDX、ECX,后面如果还有更多参数的话,就在堆栈
里面;对于类的方法,EAX 固定用于存放类实例的地址,EDX、ECX 分别存放前两个参数,其余参数进栈。在堆栈中每个元素占用 4 Bytes,而前面说了,TVarRec 中储存的
数据也是 4 Bytes,刚好一个参数在堆栈里面占一个位子,处理方便。另外,结果返回到 EAX 中。
  3、对于调用类的方法,其实有一个默认的隐藏参数 Self 作为第一个参数传入,放入 EAX 寄存器。因此我们看到的第一参数其实是第二个,因此我们处理的时候要注意。
  4、用 ObjectPascal 语法调用方法,Delphi 会自动帮我们处理参数的传递问题,而在汇编里面调用任何函数之前都需要先手动设置各参数。

因此,用内嵌汇编的办法来解决参数传递问题:如果是一个参数,放入 EDX;若为两个参数,分别放入 EDX,ECX;对多于两个参数的情况,用 参数个数 - 2 个循环依
次将后续参数进栈。然后将拥有待调方法的实例地址传入 EAX:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
function ExecuteRoutine(AObj: TObject; AName: string;
  Params: array of const): DWord;
const
  RecSize = SizeOf(TVarRec); // 循环处理参数列表时递增的字节数
var
  PFunc: Pointer;
  ParCount: DWord;
begin
  if not Assigned(AObj) then
    raise Exception.Create ('你确定传进来的是一个对象?');
  PFunc := AObj.MethodAddress(AName); // 获取方法地址
  if not Assigned(PFunc) then
    raise Exception.CreateFmt('找不到 %s 的 Method: %s', [AObj.ClassName,
      AName]);
    
  ParCount := High(Params) + 1; // 获取参数个数
 
  asm
    PUSH        ESI                 // 保存 ESI,我们待会儿要用到它
 
    MOV         ESI, Params         // ESI 指向参数表首址
    CMP         ParCount, 1         // 判断参数个数
    JB          @NoParam
    JE          @OneParam
    CMP         ParCount, 2
    JE          @TwoParams
 
  @ManyParams: // 超过两个参数
    CLD                             // 清空方向标志
    MOV         ECX, ParCount
    SUB         ECX, 2              // 循环 ParCount - 2 次
    MOV         EDX, RecSize        // EDX 依次指向每个参数的首址,每次递增 8 Bytes
    ADD         EDX, RecSize        // 跳过前两个参数
  @ParamLoop:
    MOV         EAX, [ESI][EDX]     // 用基址变址寻址方式取得一个参数
    PUSH        EAX                 // 参数进栈
    ADD         EDX, RecSize        // EDX 指向下一个参数首址
    LOOP        @ParamLoop
 
  @TwoParams: // 两个参数
    MOV         ECX, [ESI] + RecSize
 
  @OneParam: // 一个参数
    MOV         EDX, [ESI]
 
  @NoParam:
    MOV         EAX, AObj           // 传入实例地址(即,隐藏参数 Self)
    CALL        PFunc               // 调用方法
    MOV         Result, EAX         // 返回值放入 Result
 
    POP         ESI                 // 记得还原
  end;
end;

前面已经说过了,任何类型都可以塞进 4 Bytes,因此将返回值定义为 DWord,你可以根据自己的需要进行类型转换。这个办法最大限度地保护了待调方法,但也不是完全不
用修改,只有一个地方需要作出适当调整:与 DLL 中的函数返回值一样(别告诉我引用 ShareMem,那不属于本文讨论的范畴),如果要返回一个长 string,请改为 PChar
,并注意申请必要的空间。


使用示例(提醒:待调方法必须是某个类的 published 方法):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
TForm1 = class(TForm)
  Button1: TButton;
  procedure Button1Click(Sender: TObject);
private
  { Private declarations }
public
  { Public declarations }
published // 几个待调方法
  function TowInt(I, J: Integer): Integer;
  function ThreeInt(I, J, K: Integer): Integer;
  function FiveInt(X1, X2, X3, X4, X5: Integer): Integer;
  function ThreeChar(I, J, K: Char): PChar;
  function TwoStr(X, Y: string): PChar;
  function IntNBool(I: Integer; B: Boolean): Boolean;
end;
 
...
 
function ExecuteRoutine(AObj: TObject; AName: string;
  Params: array of const): DWord;
...
 
function TForm1.TowInt(I, J: Integer): Integer;
begin
  ShowMessage(Format('%d + %d', [I, J]));
  Result := I + J;
end;
 
function TForm1.ThreeInt(I, J, K: Integer): Integer;
begin
  ShowMessage(Format('%d + %d + %d', [I, J, K]));
  Result := I + J + K;
end;
 
function TForm1.FiveInt(X1, X2, X3, X4, X5: Integer): Integer;
begin
  ShowMessage(Format('%d + %d + %d + %d + %d', [X1, X2, X3, X4, X5]));
  Result := X1 + X2 + X3 + X4 + X5;
end;
 
function TForm1.ThreeChar(I, J, K: Char): PChar;
var
  Res: string;
begin
  ShowMessage(Format('%s + %s + %s', [I, J, K]));
  Res := I + J + K;
  Result := AllocMem(Length(Res) + 1);
  StrPCopy(Result, Res);
end;
 
function TForm1.TwoStr(X, Y: string): PChar;
var
  Res: string;
begin
  ShowMessage(Format('%s + %s', [X, Y]));
  Res := X + Y;
  Result := AllocMem(Length(Res) + 1);
  StrPCopy(Result, Res);
end;
 
function TForm1.IntNBool(I: Integer; B: Boolean): Boolean;
begin
  if B then
    ShowMessage(IntToStr(I) + ' and True')
  else
    ShowMessage(IntToStr(I) + ' and False');
 
  Result := B;
end;
 
procedure TForm1.Button1Click(Sender: TObject);
var
  i: Integer;
  b: Boolean;
  s: string;
begin
  i := ExecuteRoutine(Self, 'ThreeInt', [10, 23, 17]);
  ShowMessage('Result: ' + IntToStr(i));
 
  i := ExecuteRoutine(Self, 'FiveInt', [1, 2, 3, 4, 5]);
  ShowMessage('Result: ' + IntToStr(i));
 
  b := Boolean(ExecuteRoutine(Self, 'IntNBool', [10, False]));
  if b then
    ShowMessage('Result: True')
  else
    ShowMessage('Result: False');
 
  s := PChar(ExecuteRoutine(Self, 'ThreeChar', ['a', 'b', 'c']));
  ShowMessage('Result: ' + s);
 
  s := PChar(ExecuteRoutine(Self, 'TwoStr', ['hello', ' world']));
  ShowMessage('Result: ' + s);
end;

 

 

创建时间:2020.10.28  更新时间:

 

posted on   滔Roy  阅读(375)  评论(0编辑  收藏  举报

编辑推荐:
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
阅读排行:
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报

导航

点击右上角即可分享
微信分享提示