Qt应用程序在使用DLL时崩溃,如果导出的函数没有声明APIENTRY,则工作正常
Qt application crashes when using DLL, works fine if exported function doesn't declare APIENTRY
我在VisualStudio2010中构建了一个DLL项目(通过关注这篇文章)。它只包含一个功能:
extern "C" __declspec(dllexport) void APIENTRY hello()
{
std::cout << "Hello DLL.n" << std::endl;
}
然后我创建了一个Qt控制台应用程序来使用该DLL。它的main.cpp包含以下内容:
typedef bool (*f_void)(void);
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QLibrary lib("TestDll");
f_void hello = (f_void) lib.resolve(QString("hello").toLatin1());
hello();
return a.exec();
}
当我在DLL中使用APIENTRY
时,程序在调用hello()
时崩溃。不过,如果我从hello()
声明中删除APIENTRY
,效果会很好。为什么会发生这种情况?
在添加需要匹配调用约定的函数调用机制中,通过如下所述为函数指针提供正确的类型来修复该机制,调用约定会影响名称篡改。
extern "C"
防止C++将类型包含在名称中的操作方式,这样函数的重载就可以获得唯一的名称,并且可以在符号查找过程中进行区分。但这并不能完全防止损毁。例如,问题中的函数void __stdcall hello(void)
将由__declspec(dllexport)
导出为_hello@0
,其中尾随数字是参数列表中的字节数。这有助于避免调用者和被调用者在参数大小上存在分歧的情况,这在被调用者清理堆栈的__stdcall
中尤其有问题。
尽管如此,还是可以禁用名称篡改(而Win32 DLL(如gdi32.dll
和shell32.dll
)。为此,您需要一个链接器定义文件:
EXPORTS
hello
; or hello = _hello@0
链接器知道篡改规则,即使您没有明确提供,它也会在对象文件中找到被篡改的名称。
此外,当导出列在定义文件中时,代码中不再需要__declspec(dllexport)
。
MSDN上提供了更多信息。
如果通过错误类型的函数指针调用函数,则会得到未定义的行为。调用约定是类型的一部分。尝试:
typedef bool (APIENTRY *f_void)(void); // or __stdcall instead of APIENTRY
我猜你的一个头文件包含#define APIENTRY __stdcall
显式设置导出函数和函数指针的调用约定总是一个好主意。如果不这样做,您将获得当前有效的默认调用约定,这是一个特定于项目的编译器选项。
从教学意义上讲,函数和函数指针是否标记为extern "C"
也是类型的一部分。但在有DLL的Windows上,extern "C"
和extern "C++"
对调用约定有相同的作用。
- Visual Studio中的函数声明和函数定义问题
- 为什么函数声明中允许 const?
- 如果 x.h 仅由函数声明组成,为什么有必要在 x 中包含 x.h.cpp
- * 和 ** 在 C++ 函数声明中是什么意思?
- 构造函数/函数声明参数列表中的统一初始化
- 在将函数声明为友元时,尖括号的含义是什么?
- 为什么转换函数声明不需要至少一个定义类型说明符
- 如何正确编写指针函数声明?
- 在"template"和函数声明之间使用:template<typename trait> using tr = base_trait<trait> void fn(tr::t
- 为什么要将函数声明和定义放在单独的文件中
- 为什么系统日志有两个不同的函数声明?
- 我如何获取数组的大小,以便我可以从函数声明所述数组
- 使用 enable_if 在按值传递与按引用传递之间更改函数声明
- JavaScript 中的一等函数和 C++ 中的函数声明
- C++ 通过函数声明后初始化向量
- VS2017 #error: : snprintf 的宏定义与标准库函数声明冲突
- C++ 17 个友元函数声明和内联命名空间
- MSVC 2017 - 错误 - 如何将模板类 X 的模板成员函数声明为嵌套类 X::Y 的好友
- 将派生类的构造函数声明为父类的友元
- 用于从 ANSI 字符串转换为 std::basic_string <TCHAR>的正确函数声明