将记录作为函数结果从Delphi DLL传递到c++
Passing record as a function result from Delphi DLL to C++
我现在正在经历一些非常奇怪的事情。当我从c++传递一个结构体到Delphi DLL作为参数时,一切都很好。然而,一旦我想要接收记录,结果我要么得到错误的值,要么得到异常。我取消了对记录的对齐,以便传递它们应该工作!这是代码!
Delphi DLL:
TSimpleRecord = packed record
Nr1 : Integer;
Nr2 : Integer;
end;
//...
function TTest() : TSimpleRecord; cdecl;
begin
Result.Nr1 := 1;
Result.Nr2 := 201;
ShowMessage(IntToStr(SizeOf(Result)));
end;
c++调用:
#pragma pack(1)
struct TSimpleRecord
{
int Nr1;
int Nr2;
};
//...
typedef TSimpleRecord (__cdecl TestFunc)(void);
TestFunc* Function;
HINSTANCE hInstLibrary = LoadLibrary("Reactions.dll");
if (hInstLibrary)
{
Function = (TestFunc*)GetProcAddress(hInstLibrary, "TTest");
if (Function)
{
TSimpleRecord Result = {0};
Result = Function();
printf("%d - %d - %d", sizeof(Result), Result.Nr1, Result.Nr2);
cin.get();
}
}
我不知道为什么传递这个记录作为一个参数工作,而不是作为一个函数的结果!
有人能帮我吗?' 感谢PS:正如我所说的,c++和Delphi都显示记录是8字节大。
一些编译器会在寄存器中返回struct
类型(可能取决于大小),其他编译器会添加一个隐藏的额外参数来存储结果。不幸的是,看起来您正在处理两个编译器,它们不同意如何返回这些。
您应该能够通过显式地使用out
参数来避免这个问题。
procedure TTest(out Result: TSimpleRecord); cdecl;
begin
Result.Nr1 := 1;
Result.Nr2 := 201;
end;
不要忘记相应地更新c++代码。
Rudy Velthuis写过:
这表明ABCVar结构体在寄存器EDX:EAX中返回(EDX具有顶部32位,EAX具有较低的位)。Delphi根本不是这样处理记录的,即使是这么大的记录。Delphi将这样的返回类型视为额外的var参数,并且不返回任何东西(因此函数实际上是一个过程)。
[…]
Delphi返回的唯一EDX:EAX组合类型是Int64。
表明避免这个问题的另一种方法是
function TTest() : Int64; cdecl;
begin
TSimpleRecord(Result).Nr1 := 1;
TSimpleRecord(Result).Nr2 := 201;
end;
请注意,即使在c++中没有定义的情况下,Delphi也允许这种类型双关语。
Delphi不遵循返回值的平台标准ABI。标准ABI按值将返回值传递给调用者。Delphi将返回值作为一个隐式的额外var参数,在所有其他参数之后传递。文档描述了这些规则。
你可以修改你的调用代码来匹配它。在c++函数中传递一个额外的struct参数引用
typedef void (__cdecl TestFunc)(TSimpleRecord&);
如果你打算在c++端这样做,为了清晰起见,你最好在Delphi端做同样的改变。
由于Delphi不遵循返回值的平台标准,我建议您将自己限制为与其他工具兼容的类型。这意味着最多32位的整型值,指针和浮点值。
作为一般的经验法则,不要打包记录。如果这样做,就会出现不对齐,从而影响性能。对于问题中的记录,无论如何都不会有填充,因为两个字段的大小相同。
- 挂起和取消挂起一个文件DLL
- std::threads可以从Windows DLL中的全局变量创建/销毁吗?
- 导入库可以跨dll版本工作吗
- 从C++dll访问C#中的一行主要参数
- 链接到自行创建的dll失败
- 为什么使用 P/Invoke 调用 dll 时,某些计算机中的 LoadLibrary 失败?
- 在调用FreeLibrary后,释放动态链接到具有相同版本的CRT堆的DLL的内存
- 如何指定我希望我的LIB链接到的DLL文件?-Visual Studio 2019
- 如何将图像传输到c++(dll)中的缓冲区,然后在c#的缓冲区中读/写
- C++:将外部库链接到dll库
- 在 Windows 上,是否可以让 dll 在不使用 PATH 环境变量的情况下在另一个文件夹中查找依赖项?
- 不同的Visual Studio版本中缺少.dll
- 从DLL中删除类的实例
- 如何包装第三方DLL在R中使用
- 使用c#访问c++dll中带有char*参数的函数时发生AccessViolationException
- 系统.将数组移交给c#中动态加载的c++DLL时发生AccessViolationException
- 为什么导入Mixed native/CLR lib.dll的本机C++应用程序没有在Mixed lib.dll中的外部变
- 当我使用 C++ 中的 C# dll 来使用 Selenium 时,存在异常处理问题
- Python ctypes:不会按预期加载 dll
- 在 C++/CLI 中将 .NET 事件从一个 DLL 引发到另一个 DLL