调用约定的认识

想在托管代码中成功地调用非托管函数,必须根据导出非托管函数时所采用的调用约定来确定CallingConvention字段应设置的值。

CallingConvention字段用于指定调用在非托管代码中实现的方法所需的调用约定,其可取值都定义在CallingConvention枚举中。

如果不显示指定CallingConvention字段值,CLR就会默认使用Winapi。

1)采用__cdecl约定时,函数参数按照从右到左的顺序入栈,并且由调用函数者把参数弹出栈以清理堆栈。因此,实现可变参数的函数只能使用该调用约定。由于每一个使用__cdecl约定的函数都要包含清理堆栈的代码,所以产生的可执行文件大小会比较大。
2)采用__stdcall约定时,函数参数按照从右到左的顺序入栈,被调用的函数在返回前清理传送参数的栈,函数参数个数固定。由于函数体本身知道传进来的参数个数,因此被调用的函数可以在返回前用一条ret n指令直接清理传递参数的堆栈。

3)采用Winapi时,此成员实际上不是调用约定,而是使用默认平台调用约定。例如,在Windows平台上默认为StdCall,而在WindowsCE.NET(.NET Compact Freamework)上默认为Cdecl。

调用约定一个作用是决定谁负责清理堆栈。另一个作用在于它控制了非托管函数的名称重整,影响着非托管函数被导出的实际名称。例如,当一个非托管函数被导出时,使用了extern “C”和_stdcall进行定义,其函数名就会别重整成“下划线+原函数名+@+所有函数参数的总字节数”格式。

                                                                                                                                                                                                                                                        摘抄自《精通.NET互操作》

你可能感兴趣的:(调用约定的认识)