我知道如何在 Unix-ish 上下文中链接库:如果我正在使用.a
or文件,我使用和为 libMylib 添加.so
指定根搜索目录。-L/my/path/to/lib/
-lMyLib
但是如果我有
- 一个
.dll
(例如在Windows\System32
目录中)? - 一个
.dll
(在Windows\System32
)和一个.lib
(其他地方)?
这些 DLL 来自其他方;我无权访问他们的源代码 - 但可以访问相应的包含文件,我可以根据这些文件进行编译。
如果您可以链接.lib
到 Cygwin 或 MinGW 中的 a,那么您可以(间接)链接到 DLL。
在 MSVC 世界中,与 DLL 一起创建导入库并不罕见。它是一个静态库 ( .lib
),加载 DLL 并包装 DLL 的接口。您只需调用(静态)导入库中的包装函数,并让导入库完成所有与 DLL 相关的事情。
.lib
可以在 Cygwin 或 MinGW 中链接文件。例子:
g++ -o myprg myprg.o -lShlwapi
这与 Shlwapi.lib 链接。(库必须在本地目录或链接器的库路径中。)
链接到 DLL 的导入库的工作方式相同。
注意 1:请记住不同的 ABI 和名称管理。但是,在 DLL 或 LIB 文件中调用普通 C 函数在大多数情况下都可以工作。
注意 2:请记住,g++ 要求以正确的顺序指定库。
@einpoklum 将我的评论转换为答案:@n.18e9 是正确的,因为您必须使用 lib 文件的完整路径名,而不使用任何 -L 或 -l 选项。
g++ -o foo.exe foo.o c:\something\somethingelse\some.lib
. 您也可以直接链接到 Windows DLL 文件g++ -o foo.exe foo.o c:\something\somethingelse\some.dll
。
重要- 确保您链接到为64 位平台(在 MSVC 目标 X64,而不是 Win32)生成的 lib 文件(和关联的 dll)。
好的,你想要一个例子,我们走吧。
下面是两个使用 gcc/g++ 链接到导出纯 C 函数的 Windows 本机 DLL 的示例(在 Windows 10 上使用 x86_64-w64-mingw32/8.3.0)。
我以我自己的免费 xmlsq 库为例https://www.cryptosys.net/xmlsq。您可以下载核心原生 DLL 和下面引用的所有源代码。确保使用 64 位 DLL。
本机 Windows DLLdiXmlsq.dll
完全用纯 C 代码编写,并导出简单的 C 函数(外部“C”)。特别是,对于这个例子,它导出了一个XMLSQ_Gen_Version
返回整数值的函数。DLL 是使用面向 X64 平台的 MSVC 12.0 编译的。MSVC 生成的关联库文件是diXmlsq.lib
.
我应该补充一点,这个 DLL 的工作方式与 Windows“Win32 API”DLL 完全相同,因此这里的说明应该适用于标准 Windows 库Windows\System32
(再次确保链接到 64 位版本)。
示例 1。一个普通的 C 接口。
这两个命令在我的系统上编译时没有警告:
> gcc -o test-ver test-ver.c "C:\fullpath\to\x64\diXmlsq.lib"
> gcc -o test-ver test-ver.c "C:\fullpath\to\x64\diXmlsq.dll"
diXmlsq.dll
使用以下定义文件编译。(您也可以使用__declspec(dllexport)
)
参考:https ://docs.microsoft.com/en-us/cpp/build/exporting-from-a-dll?view=msvc-160
diXmlsq.def
LIBRARY "diXmlsq"
EXPORTS
XMLSQ_Gen_Version
diXmlsq.h
- C 接口diXmlsq.dll
#ifdef __cplusplus
extern "C" {
#endif
long __stdcall XMLSQ_Gen_Version(void);
#ifdef __cplusplus
}
#endif
在纯 C 程序中调用核心函数:
test-ver.c
#include <stdio.h>
#include "diXmlsq.h"
int main(void)
{
long n;
n = XMLSQ_Gen_Version();
printf("Version = %ld\n", n);
return 0;
}
示例 2。一个 C++ 接口。
这两个命令都使用 g++ 编译而没有警告。
> g++ -o test-simple test-simple.cpp xmlsq.cpp "C:\fullpath\to\x64\diXmlsq.lib"
> g++ -o test-simple test-simple.cpp xmlsq.cpp "C:\fullpath\to\x64\diXmlsq.dll"
C++ 接口的想法是成为普通 C 库的接口,使用更方便的 STL 类型,如std::string
和std::vector
. 为简单起见,我们将仅演示该Gen::Version
方法。
C++ 代码摘录如下:
test-simple.cpp
- 一个测试 C++ 程序。
#include <iostream>
#include "xmlsq.hpp"
int main()
{
std::cout << "xmlsq::Gen::Version=" << xmlsq::Gen::Version() << std::endl;
}
xmlsq.hpp
- C++ 接口
namespace xmlsq
{
class Gen {
private:
Gen() {} // Static methods only, so hide constructor.
public:
/** Get version number of core diXmlsq DLL. */
static int Version();
};
}
xmlsq.cpp
- C++ 实现。
#include "diXmlsq.h"
#include "xmlsq.hpp"
namespace xmlsq
{
int Gen::Version() {
int n = XMLSQ_Gen_Version();
return n;
}
}
例 3。试图错误地链接到 32 位库。
> gcc -o test-ver test-ver.c "C:\fullpath\to\Win32\diXmlsq.lib"
C:/Strawberry/c/bin/../lib/gcc/x86_64-w64-mingw32/8.3.0/../../../../x86_64-w64-mingw32/bin/ld.exe:
C:\Users\user\AppData\Local\Temp\cce27Dhl.o:test-ver.c:(.text+0xe):
undefined reference to `XMLSQ_Gen_Version'
collect2.exe: error: ld returned 1 exit status