尝试编译此cfgparser example。
$ g++ example.cc -lcfgparser
: In function `main':
example.cc:(.text+0x6b): undefined reference to `ConfigParser_t::readFile(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)'
example.cc:(.text+0x160): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*) const'
example.cc:(.text+0x2d9): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, int*) const'
example.cc:(.text+0x43c): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, double*) const'
example.cc:(.text+0x5b1): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, bool*) const'
example.cc:(.text+0x78c): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > >*) const'
example.cc:(.text+0xa15): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*) const'
example.cc:(.text+0xba2): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*) const'
example.cc:(.text+0xd15): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*) const'
example.cc:(.text+0xe7f): undefined reference to `ConfigParser_t::getValue(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*) const'
collect2: error: ld returned 1 exit statu
但显然这些符号在那里:
$ nm -gC /usr/lib/libcfgparser.so
000000000003710 T ConfigParser_t::readFile(std::string const&)
0000000000004880 T ConfigParser_t::ConfigParser_t(std::string const&)
00000000000024c0 T ConfigParser_t::ConfigParser_t()
0000000000004ad0 T ConfigParser_t::ConfigParser_t(std::string const&)
00000000000024a0 T ConfigParser_t::ConfigParser_t()
0000000000004d20 T ConfigParser_t::getOptions(std::string const&) const
00000000000028d0 T ConfigParser_t::getSections() const
0000000000002ff0 T ConfigParser_t::getValue(std::string, std::string, bool*) const
0000000000002de0 T ConfigParser_t::getValue(std::string, std::string, double*) const
0000000000002bd0 T ConfigParser_t::getValue(std::string, std::string, int*) const
00000000000027d0 T ConfigParser_t::getValue(std::string, std::string, std::string*) const
0000000000003500 T ConfigParser_t::getValue(std::string, std::string, std::vector<std::string, std::allocator<std::string> >*) const
与SO上的其他类似问题不同,这与链接顺序无关,因为仅链接了一个目标文件。我想念的东西吗?
最佳答案
But clearly those symbols are there:
显然,他们不是。仔细看看,您会看到
您的报告的签名之间没有匹配项
链接器未定义,并通过以下方式从库中报告
nm
。标准 header
<string>
将std::string
定义为typedef,用于:std::basic_string<char, std::char_traits<char>, std::allocator<char>>
从cxx11 ABI开始的
<string>
的GCC实现(GCC 4.7.0)定义
std::string
为typedef用于:std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>
std::__cxx11
命名空间包含其二进制实现为符合cxx11 ABI。
该类型定义意味着包含Standard header
<string>
的C++转换单元将不会在GCC> = 4.7的情况下,将其编译为目标代码,其中包含分解为
std::string
的符号。如果二进制文件(如
libcfgparser.so
)包含一个符号分解为std::string
,则无论该符号在构建它的源中可能已经引用过,它都不会引用
std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>
,并且不能与使用
std::string
定义编译的其他二进制文件链接。您的
libcfgparser.so
是libcfgparser0_1.1.2_amd64.deb
从cfgparser Sourceforce project
(最近更新于三年前)。我从这个事实猜测到
使用与该软件包安装的库链接的相同
example.cc
程序获取链接错误。该错误的解释通过以下方式揭示:
/usr/lib$ strings -a libcfgparser.so | grep "GCC: ("
GCC: (Debian 4.3.2-1.1) 4.3.2
...
告诉我们该库是使用GCC 4.3.2构建的-在cxx11 ABI之前。
std::string
输出中显示的nm
是cxx11之前的分词缩写的
std::basic_string<char, std::char_traits<char>, std::allocator<char>>
。此
libcfgparser.so
与您当前的GCC不兼容,因此您无法将其链接到使用该编译器生成的程序。你能做的是
从源代码包
libcfgparser
构建libcfgparser-1.1.2.tar.bz2
,使用当前的编译器,然后将程序与您拥有的库链接
建立自己。
那行得通,但并非毫不费力。首先,您需要修复损坏的
和过时的源代码包自动工具创建有效的
./configure
脚本。这并不能使人对包维护者的熟练程度有任何放心的印象。
加上该软件包的源代码(版权为2008),具有业余质量。如果
您需要的是用于INI样式文件的C++解析器,然后
boost::property_tree::ini_parser
将是转到解决方案。参见this answer
关于c++11 - 链接器找不到符号,但是它们在那里?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41134190/