MinGW + Boost:未定义对"WSAStartup@8"的引用

MinGW + Boost: undefined reference to `WSAStartup@8'

本文关键字:WSAStartup@8 引用 未定义 Boost MinGW      更新时间:2023-10-16

下面是我执行的内容

D:Just4Funsoftware>make -f Makefile.hands stest
g++.exe -g -D_WIN32_WINNT=0x0601 -ID:Boostboost_1_51_0 -LD:Boostboost_1_51_0stagelib -LD:MinGWlib -lboost_system-mgw46-d-1_51 -lboost_filesystem-mgw46-d-1_51 -lboost_iostreams-mgw46-d-1_51 -lws2_32 -lwsock32 -o TestSerial.exe TestSerial.cpp

以下是结果

C:UsersolegAppDataLocalTempccXpC7RG.o: In function 'ZN5boost4asio6detail17winsock_init_base7startupERNS2_4dataEhh': D:/Boost/boost_1_51_0/boost/asio/detail/impl/winsock_init.ipp:39: undefined reference to 'WSAStartup@8'C:UsersolegAppDataLocalTempccXpC7RG.o: In function 'ZN5boost4asio6detail17winsock_init_base7cleanupERNS2_4dataE': D:/Boost/boost_1_51_0/boost/asio/detail/impl/winsock_init.ipp:48: undefined reference to 'WSACleanup@0' collect2: ld returned 1 exit status make: *** [stest] Error 1

以下是检查有关上述符号的库的结果(我未能以适当的方式格式化它)

'd:\MinGW\lib>nm -C libws2_32.a | grep WSAStartup
00000000 T WSAStartup@8
00000000 I imp_WSAStartup@8

d:\MinGW\lib>nm -C libws2_32.a | grep WSACleanup
00000000 T WSACleanup@0
00000000 我imp_WSACleanup@0

d:\MinGW\lib>nm -C libwsock32.a | grep WSACleanup
00000000 T WSACleanup@0
00000000 我imp_WSACleanup@0

d:\MinGW\lib>nm -C libwsock32.a | grep WSAStartup
00000000 T WSAStartup@8
00000000 我_WSAStartup@8'

如我们所见,函数库中存在符号,并且库位于相应的 -L 命令行选项引用的位置。
那么,为什么是"未定义的引用"????

这是一个非常常见的错误,有一个非常简单的解决方案:将链接的库放在源文件和目标文件之后


如果对象(或源)文件 A 依赖于库 B,则链接时 A 必须在命令行上的 B 之前。这是因为 GNU(可能还有其他)链接器如何解决符号依赖关系。

如果以相反(和错误)的方式做

g++ -lB A.cpp

然后,当链接器加载库 B 时,对库中的任何函数都没有依赖关系,因此链接器只是丢弃该库。

所以正确的方法是把库放在最后

g++ A.cpp -lB

这就是为什么许多示例总是将库放在命令行的最后

所以对于问题的命令行,它应该是

g++.exe -g -D_WIN32_WINNT=0x0601 -ID:Boostboost_1_51_0 
    -LD:Boostboost_1_51_0stagelib -LD:MinGWlib 
    -o TestSerial.exe TestSerial.cpp 
    -lboost_system-mgw46-d-1_51 -lboost_filesystem-mgw46-d-1_51 -lboost_iostreams-mgw46-d-1_51 -lws2_32 -lwsock32

此顺序对于库间依赖项也很重要。如果库 L1 依赖于库 L2,则在命令行上 L1 必须位于 L2 之前。