__declspec(dllimport)的真正含义是什么

what does __declspec(dllimport) really mean?

本文关键字:是什么 declspec dllimport      更新时间:2023-10-16

我看到了这样的Qt源代码:

class Q_CORE_EXPORT QBasicAtomicInt
{
public:
...
};

哪个Q_CORE_EXPORT宏定义如下:

define Q_DECL_IMPORT __declspec(dllimport)

那么__declspec(dllimport)到底是什么意思呢?

__declspec是Microsoft特定的属性,允许您指定存储类信息
(Nitchicker's Corner:然而,许多其他编译器供应商(如GCC)现在支持这种语言扩展,以与针对Microsoft编译器编写的已安装代码库兼容。有些甚至提供了额外的存储类属性。)

其中两个可以指定的存储类属性是dllimportdllexport。它们向编译器指示函数或对象是从DLL(分别)导入或导出的。

更具体地说,它们定义了DLL到客户端的接口,而不需要模块定义(.DEF)文件。大多数人发现使用这些语言扩展比创建DEF文件容易得多。

由于显而易见的原因,__declspec(dllimport)__declspec(dllexport)通常彼此配对。使用dllexport将符号标记为从DLL导出,使用dllimport将导出的符号导入到另一个文件中。

正因为如此,而且在编译DLL时和使用DLL接口的客户端代码中通常都使用相同的头文件,所以定义一个在编译时自动解析为适当属性说明符的宏是一种常见的模式。例如:

#if COMPILING_DLL
    #define DLLEXPORT __declspec(dllexport)
#else
    #define DLLEXPORT __declspec(dllimport)
#endif

然后用DLLEXPORT标记所有应该导出的符号。

据推测,这就是Q_CORE_EXPORT宏所做的,解析为Q_DECL_IMPORTQ_DECL_EXPORT

__declspec(dllimport)是一个存储类说明符,它告诉编译器在外部DLL中定义了函数、对象或数据类型。

函数、对象或数据类型是从具有相应__declspec(dllexport)的DLL中导出的。

__declspec(dllexport)告诉编译器通知链接器这些符号需要放在导出表中(编译.dll时),并将这些符号放在导入库.lib中。当编译与.dll链接的程序时,__declspec(dllimport)告诉编译器生成一个rip-relativememory-indirect调用(链接器将填充解析以指向导入表),而不是对未定义函数的通常的相对直接指令(由于无法修改指令,链接器插入thunk的相对地址,然后创建thunk,在其中放置指向导入表中函数指针的rip相对内存间接跳转)。这是对代码大小和速度的优化。是导入库.lib告诉链接器哪些符号由.dll导出,并用作根据那些符号与匹配的外部符号表项的交集创建导入表的指南,并在.text段中创建任何必要的thunk。

https://learn.microsoft.com/en-us/cpp/build/importing-function-calls-using-declspec-dllimport?view=vs-2019https://learn.microsoft.com/en-us/cpp/build/importing-data-using-declspec-dllimport?view=vs-2019https://stackoverflow.com/a/4490536/7194773

这意味着函数的定义在动态库中。有关更多详细信息和示例,请参阅文档。