这是一个相当简单的应用程序的设置,我无法正确链接。基本上是单个源文件main.cpp
,带有静态库libbreakpad_client.a
的目录,以及breakpad
中库的标题。
$ ls .
main.cpp libs breakpad
$ ls libs
libbreakpad_client.a
以下是main.cpp
的来源,从the Google Breakpad documentation逐字逐句。
#include "client/linux/handler/exception_handler.h"
void crash()
{
volatile int* a = (int*)(NULL);
*a = 1;
}
static bool dumpCallback(const google_breakpad::MinidumpDescriptor& descriptor, void* context, bool succeeded)
{
printf("Dump path: %s\n", descriptor.path());
return succeeded;
}
int main()
{
google_breakpad::MinidumpDescriptor descriptor("/tmp");
google_breakpad::ExceptionHandler eh(descriptor, 0, dumpCallback, 0, true, -1);
crash();
return 0;
}
但是当我尝试编译它时,链接器抱怨它找不到ExceptionHandler
构造函数和析构函数。
$ g++ -Ibreakpad/src -lbreakpad_client -Llibs -o main main.cpp
/tmp/ccDDGnQk.o: In function `main':
main.cpp:(.text+0xed): undefined reference to `google_breakpad::ExceptionHandler::ExceptionHandler(google_breakpad::MinidumpDescriptor const&, bool (*)(void*), bool (*)(google_breakpad::MinidumpDescriptor const&, void*, bool), void*, bool, int)'
main.cpp:(.text+0x106): undefined reference to `google_breakpad::ExceptionHandler::~ExceptionHandler()'
collect2: error: ld returned 1 exit status
但如果libbreakpad_client.a
中出现look at the symbols,我们会看到函数完全按照预期存在
$ nm -an libs/libbreakpad_client.a | c++filt | grep "ExceptionHandler("
000000000000076c T google_breakpad::ExceptionHandler::ExceptionHandler(google_breakpad::MinidumpDescriptor const&, bool (*)(void*), bool (*)(google_breakpad::MinidumpDescriptor const&, void*, bool), void*, bool, int)
000000000000076c T google_breakpad::ExceptionHandler::ExceptionHandler(google_breakpad::MinidumpDescriptor const&, bool (*)(void*), bool (*)(google_breakpad::MinidumpDescriptor const&, void*, bool), void*, bool, int)
0000000000000972 T google_breakpad::ExceptionHandler::~ExceptionHandler()
0000000000000972 T google_breakpad::ExceptionHandler::~ExceptionHandler()
那么为什么链接器会抱怨这些是undefined reference
?
答案 0 :(得分:6)
编译器的标志顺序很重要。您需要在源文件名之后放置-l
标志:
$ g++ -Ibreakpad/src -Llibs -o main main.cpp -lbreakpad_client
在您编写此选项的命令中,它会有所不同;链接器按照指定的顺序搜索和处理库和目标文件。因此,
foo.o -lz bar.o
在文件z
之后但 {{1}之前搜索库foo.o
}} 即可。如果bar.o
引用bar.o
中的功能,则可能无法加载这些功能。
答案 1 :(得分:3)
链接线的顺序很重要!只有在参数列表中显示库时,才会搜索一次库。如果此时没有未解析的符号,则不会发生任何其他符号。将命令更改为
g++ -Ibreakpad/src -Llibs -o main main.cpp -lbreakpad_client