本文详解如何在go中通过cgo正确链接并调用msvc编译的windows动态链接库(dll),解决常见“undefined reference”链接错误,避免绕行syscall包。
在Go
中调用Windows DLL时,一个常见误区是试图将DLL文件(如 MyModule.dll)直接作为静态链接目标写入 #cgo LDFLAGS。实际上,cgo的 LDFLAGS 仅用于链接时(link-time)解析符号,而Windows DLL需通过导入库(.lib)或动态加载机制支持符号解析——直接指定 .dll 文件路径无法满足链接器对存根符号(import stubs)的需求。
MSVC生成DLL时,默认会同时输出一个同名的 .lib 文件(例如 MyModule.lib),该文件包含DLL导出函数的链接存根。你需要:
修正后的Go代码如下:
//#cgo CFLAGS: -IC:/Repos/Module/include //#cgo LDFLAGS: -LC:/Repos/Module/lib -lMyModule //#includeimport "C" func main() { nRet := C.moduleImpl_len() // ... }
⚠️ 注意事项:-L 指定库搜索目录(路径需存在且含 MyModule.lib);-lMyModule 告知链接器查找 libMyModule.lib 或 MyModule.lib(Windows下优先匹配后者);不要在 LDFLAGS 中写 .dll 文件路径——这会导致链接器忽略它,因为DLL本身不提供链接期符号定义;若仅拥有 .dll 而无 .lib,可使用 dumpbin /exports MyModule.dll > exports.txt 提取符号,再用 lib /def:MyModule.def /out:MyModule.lib 手动生成导入库(需先编写DEF文件)。
确保头文件声明、DLL导出与Go调用三者完全一致:
extern "C" __declspec(dllimport) int moduleImpl_len(void);
若无法获取或生成 .lib,可退回到 syscall + LoadLibrary/GetProcAddress 方式,但会失去类型安全和编译期检查。本文推荐优先采用 -l 链接方式,兼顾简洁性与可靠性。
总结:Go调用Windows DLL的核心在于链接阶段依赖导入库(.lib),而非DLL本身;正确配置 #cgo LDFLAGS: -L