8

下面是我执行的

D:\Just4Fun\software\>make -f Makefile.hands stest
g++.exe -g -D_WIN32_WINNT=0x0601 -ID:\Boost\boost_1_51_0 -LD:\Boost\boost_1_51_0\stage\lib -LD:\MinGW\lib -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:\Users\oleg\AppData\Local\Temp\ccXpC7RG.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:\Users\oleg\AppData\Local\Temp\ccXpC7RG.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 I imp _WSACleanup@0

d:\MinGW\lib>nm -C libwsock32.a | grep WSACleanup
00000000 T WSACleanup@0
00000000 I imp _WSACleanup@0

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

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

4

1 回答 1

16

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


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

如果以相反(和错误)的方式进行操作

g++ -lB A.cpp

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

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

g++ A.cpp -lB

这就是许多示例总是将库放在命令行最后的原因。

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

g++.exe -g -D_WIN32_WINNT=0x0601 -ID:\Boost\boost_1_51_0 \
    -LD:\Boost\boost_1_51_0\stage\lib -LD:\MinGW\lib \
    -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 之前。

于 2012-09-18T08:04:05.313 回答