链接libthrift.so出现带“__cxx11”的undefined symbol的问题解决

一、问题描述

项目中一个C++程序要读写hbase的数据,按thrift接口规范编写好代码,在windows平台该程序运行正常。但在移植到linux平台后,在编译链接时一直报undefined symbol错误,即使采用其它技术手段绕过这个错通过编译链接,运行时仍会出错。

经检查,出错是因为一个模块(lib_hbase_reader.so)中调用的三个接口与libthrift-0.10.0.so中提供的接口不一致引起的,使用ldd得到的信息为:

$ldd -r libhbase_reader.so
...
libthrift-0.10.0.so => /usr/local/lib64/libthrift-0.10.0.so (0x00007f91ea5fb000)
...

undefined symbol: _ZN6apache6thrift5async25TConcurrentClientSyncInfo10getPendingERNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERNS0_8protocol12TMessageTypeERi (./lib_hbase_reader.so)
  undefined symbol: _ZN6apache6thrift9transport7TSocketC1ERKNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEi (./lib_hbase_reader.so)
  undefined symbol: _ZN6apache6thrift5async25TConcurrentClientSyncInfo13updatePendingERKNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEENS0_8protocol12TMessageTypeEi (./lib_hbase_reader.so)

以其中的getPending为例,在libthrift-0.10.0.so中提供的接口形式却是:

$ nm -a /usr/local/lib/libthrift-0.10.0.so | grep getPending
0000000000046b10 T _ZN6apache6thrift5async25TConcurrentClientSyncInfo10getPendingERSsRNS0_8protocol12TMessageTypeERi

二、出错原因

从出错信息看,貌似是两个so对某个类型产生了分歧,一个按__cxx11...,而另一个按基本类型。

看起来简单,解决之路却异常艰难,一度到了山穷水尽的地步。隔了一段时间后不得不重新面对,又经历了很多艰难险阻,终于找到原因,原来与编译所用的gcc版本密切相关:

  • lib_hbase.reader.so与libthrift-0.10.0.so不是同一个版本的gcc编译生成的,前一个因最近生成,使用的是gcc 7.3.0,而后一个是从svn库拿到的老版本,使用的是gcc 4.8.5。
  • 这就over了?远远没有,运行程序的服务器如果不是同一版本,运行时一样出错!必须将运行服务器的gcc也升级到相同版本。

解决过程中还出现了种种小陷阱,不过最终都解决了。

三、关键步骤之:升级gcc

Linux默认的gcc版本是4.8.5,如果编译某个模块时用的是高版本gcc,就需将所有相关Linux服务器的gcc都进行升级。这里以7.3.0为例。

# wget http://mirrors.kernel.org/gun/gcc/gcc-7.3.0/gcc-7.3.0.tar.gz
# tar -xvf gcc-7.3.0.tar.gz
# cd gcc-7.3.0
# ./contrib/download_prerequisites
# mkdir build
# cd build
# ../configure --enable-checking=release --enable-languages=c,c++ --disable-multilib
# make
# make install

make的时间会很长,可用make -j4开四个任务加快速度。/contrib/download_prerequisites对运行机不是必须的,这对内网环境是个好消息。

升级完后,还需要更新/usr/bin/目录下的c++, g++, gcc*等文件,以及libstdc++.so.6。

# mv /usr/bin/c++ /usr/bin/c++.bak
# mv /usr/bin/g++ /usr/bin/g++.bak
# mv /usr/bin/gcc /usr/bin/gcc.bak
# mv /usr/bin/gcc-ar /usr/bin/gcc-ar.bak
# mv /usr/bin/gcc-nm /usr/bin/gcc-nm.bak
# mv /usr/bin/gcc-ranlib /usr/bin/gcc-ranlib.bak
# ln -s /usr/local/bin/c++ /usr/bin/c++
# ln -s /usr/local/bin/g++ /usr/bin/g++
# ln -s /usr/local/bin/gcc /usr/bin/gcc
# ln -s /usr/local/bin/gcc-ar /usr/bin/gcc-ar
# ln -s /usr/local/bin/gcc-nm /usr/bin/gcc-nm
# ln -s /usr/local/bin/gcc-ranlib /usr/bin/gcc-ranlib

# rm /lib64/libstdc++.so.6
# ln -s /usr/local/lib64/libstdc++.so.6.0.24 /lib64/libstdc++.so.6

这里采用的是“旧文件改名+建软连接”的方式,还有一种方式是“拷贝新文件覆盖”的方式。

完成后查看gcc版本:

# gcc -v

四、关键步骤之:重新编译libthrift

如果gcc已升级,必须重新编译libthrift,假设为0.10.0版,已有源码包,不必再下载。

# tar -zxvf libthrift-0.10.0.tar.gz
# cd  libthrift-0.10.0
# ./configure --with-cpp --with-boost --without-python CPPFALGS=-std=c++11
# ./make
# ./make install

特别注意:configure时必须加“CPPFALGS=-std=c++11”选项,否则错误依旧。在用gcc 7.3.0编译时,会出很多警告信息,但貌似没影响。

libthrift依赖的libboost则不需要重新编译。

原文地址:https://www.cnblogs.com/wggj/p/10333394.html

时间: 2024-08-05 14:13:11

链接libthrift.so出现带“__cxx11”的undefined symbol的问题解决的相关文章

[速记]python: symbol lookup error: /usr/lib/x86_64-linux-gnu/libatk-1.0.so.0: undefined symbol: g_log_structured_standard问题解决

python: symbol lookup error: /usr/lib/x86_64-linux-gnu/libatk-1.0.so.0: undefined symbol: g_log_structured_standard https://packages.debian.org/sid/amd64/libatk1.0-0/download sudo dpkg -i *.deb Done! 原文地址:https://www.cnblogs.com/DataNerd/p/9094972.ht

libphp5.so: undefined symbol: OnUpdateLong

由于第一次安装后效果没有出来,后来又安装了一次,发现安装之后无法启动apache,报错:Cannot load /usr/lib/apache2/modules/libphp5.so into server:/usr/lib/apache2/modules/libphp5.so: undefined symbol: OnUpdateLong 问题解决: 删除已经安装的php文件,以及解压好的php-5.4.45文件夹,然后重新解压php-5.4.45.tar.gz文件,重新进行安装即可解决.

libsvn_subr-1.so.0: undefined symbol: apr_atomic_xchgptr 故障解决

源码编译安装完成之后,查看svn的安装版本会报以下错误 svn: symbol lookup error: /usr/local/subversion/lib/libsvn_subr-1.so.0: undefined symbol: apr_atomic_xchgptr 问题是出在了libsvn_subr-1.so.0库文件上,那就先查看一下它所依赖的动态连接库信息吧 ldd /usr/local/subversion/lib/libsvn_subr-1.so.0 linux-vdso.so.

ssh升级以及ssh: symbol lookup error: ssh: undefined symbol: EVP_aes_128_ctr错误处理

1.解压安装openssl包:(不能卸载openssl,否则会影响系统的ssl加密库文件,除非你可以做两个软连接libcryto和libssl) # tar -zxvf openssl-1.0.1.tar.gz # cd openssl-1.0.1 #./config -fPIC threads shared # make # make test # make install # mv /usr/bin/openssl /usr/bin/openssl.OFF # mv /usr/include

centos使用yum安装软件的时候出现了undefined symbol: CRYPTO_set_locking_callback

1.问题 在CentOS下使用yum安装软件,结果出现了下面的错误提示: # yum installThere was a problem importing one of the Python modulesrequired to run yum. The error leading to this problem was: /usr/lib/python2.6/site-packages/pycurl.so: undefined symbol: CRYPTO_set_locking_call

编译时出现莫名其妙的undefined symbol的可能原因

前言 公司里需要开发多个动态库链接在一起的程序,然而经常在一些地方出现莫名其妙的undefined symbol问题 解决方案 1.检查Makefile/CMakeList.txt,查看是否包含所有需要包含的文件/文件夹. 错误原因:有时候由于新添加了一个文件/文件夹,而该文件/文件夹又没有被Makefile/CMakeList.txt扫描到,这时候就会在实际运行时出现undefined symbol,原因是编译时找到了对应的头文件,却在链接时未找到需要的头文件. 2.查看对应函数在头文件和Cp

stm32f405xx.h头文件的问题Undefined symbol IS_TIM_BREAK_INSTANCE

1. 在实际使用过程中发现,编译工程中,出了个错误Undefined symbol IS_TIM_BREAK_INSTANCE 经过查找,发现有两个stm32f405xx.h,其中一个是,安装的器件包里面带的 2. 第二个是,STM32F4的HAL库里面的,我们实际应该用这个,但是有时候吧,KEIL 会先找到第一个,那么编译就会出现上面的问题 原文地址:https://www.cnblogs.com/429512065qhq/p/10889445.html

Centos下Apache遇到/usr/lib64/libnsssysinit.so: undefined symbol: PR_GetEnvSecure错误

错误详情: /usr/local/apache/bin/httpd: symbol lookup error: /usr/lib64/libnsssysinit.so: undefined symbol: PR_GetEnvSecure 错误产生原因: Centos官方发布了一个新版本的NSS包,如果进行更新就会产生这样的错误 解决方法: 重启Apache(开启了php-fpm也需要重启) 参考文章:Httpd - Symbol Lookup Errors

look up error: undefined symbol:......解决过程

今天遇到个很郁闷的问题,运行程序时报错:look up error: undefined symbol:... 首先运行时报错找不到,一般就是动态库里没有这个符号.所以我用 nm命令 并用管道结合grep命令(nm .so | grep 'pattern') 查找了下那个未定义符号.结果发现,动态库里有这个符号,这里T表示是在代码段. 然后我怀疑可能加载的不是这个库,可能是老版本的库,于是我又用 ldd 命令执行可执行文件,查看运行时程序加载了哪些库,(这里也可以用管道+grep),结果发现加载