为什么链接器找不到静态库中存在的符号?

时间:2013-09-25 18:30:35

标签: c++ linker static-libraries

这是一个相当简单的应用程序的设置,我无法正确链接。基本上是单个源文件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

2 个答案:

答案 0 :(得分:6)

编译器的标志顺序很重要。您需要在源文件名之后放置-l标志

$ g++ -Ibreakpad/src -Llibs -o main main.cpp -lbreakpad_client

来自GCC documentation

  

在您编写此选项的命令中,它会有所不同;链接器按照指定的顺序搜索和处理库和目标文件。因此, 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