Ubuntu 下配置protobuf以及ldconfig

详解命令-ldconfig

2018-01-06 16:29 操作系统

导读ldconfig命令的用途主要是在默认搜寻目录/lib和/usr/lib以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索出可共享的动态链接库(格式如lib*.so*),进而创建出动态装入程序(ld.so)所需的连接和缓存文件。缓存文件默认为/etc/ld.so.cache,此文件保存已排好序的动态链接库名字列表,为了让动态链接库为系统所共享,需运行动态链接库的管理命令ldconfig,此执行程序存放在/sbin目录下。

 

ldconfig通常在系统启动时运行,而当用户安装了一个新的动态链接库时,就需要手工运行这个命令。

语法

ldconfig [-v|--verbose] [-n] [-N] [-X] [-f CONF] [-C CACHE] [-r ROOT] [-l] [-p|--print-cache] [-c FORMAT] [--format=FORMAT] [-V] [-?|--help|--usage] path...

选项

-v或--verbose:用此选项时,ldconfig将显示正在扫描的目录及搜索到的动态链接库,还有它所创建的连接的名字。

-n:用此选项时,ldconfig仅扫描命令行指定的目录,不扫描默认目录(/lib、/usr/lib),也不扫描配置文件/etc/ld.so.conf所列的目录。

-N:此选项指示ldconfig不重建缓存文件(/etc/ld.so.cache),若未用-X选项,ldconfig照常更新文件的连接。

-X:此选项指示ldconfig不更新文件的连接,若未用-N选项,则缓存文件正常更新。

-f CONF:此选项指定动态链接库的配置文件为CONF,系统默认为/etc/ld.so.conf。

-C CACHE:此选项指定生成的缓存文件为CACHE,系统默认的是/etc/ld.so.cache,此文件存放已排好序的可共享的动态链接库的列表。

-r ROOT:此选项改变应用程序的根目录为ROOT(是调用chroot函数实现的)。选择此项时,系统默认的配置文件/etc/ld.so.conf,实际对应的为ROOT/etc/ld.so.conf。

如用-r /usr/zzz时,打开配置文件/etc/ld.so.conf时,实际打开的是/usr/zzz/etc/ld.so.conf文件。用此选项,可以大大增加动态链接库管理的灵活性。

-l:通常情况下,ldconfig搜索动态链接库时将自动建立动态链接库的连接,选择此项时,将进入专家模式,需要手工设置连接,一般用户不用此项。

-p或--print-cache:此选项指示ldconfig打印出当前缓存文件所保存的所有共享库的名字。

-c FORMAT 或 --format=FORMAT:此选项用于指定缓存文件所使用的格式,共有三种:old(老格式),new(新格式)和compat(兼容格式,此为默认格式)。

-V:此选项打印出ldconfig的版本信息,而后退出。

-? 或 --help 或 --usage:这三个选项作用相同,都是让ldconfig打印出其帮助信息,而后退出。

ldconfig几个需要注意的地方:

1,往/lib和/usr/lib里面加东西,是不用修改/etc/ld.so.conf的,但是完了之后要调一下ldconfig,不然这个library会找不到。

2,想往上面两个目录以外加东西的时候,一定要修改/etc/ld.so.conf,然后再调用ldconfig,不然也会找不到。比如安装了一个mysql到/usr/local/mysql,mysql有一大堆library在/usr/local/mysql/lib下面,这时就需要在/etc/ld.so.conf下面加一行/usr/local/mysql/lib,保存过后ldconfig一下,新的library才能在程序运行时被找到。

3,如果想在这两个目录以外放lib,但是又不想在/etc/ld.so.conf中加东西(或者是没有权限加东西)。那也可以,就是export一个全局变量LD_LIBRARY_PATH,然后运行程序的时候就会去这个目录中找library。一般来讲这只是一种临时的解决方案,在没有权限或临时需要的时候使用。

4,ldconfig做的这些东西都与运行程序时有关,跟编译时一点关系都没有。编译的时候还是该加-L就得加,不要混淆了。

5,总之,就是不管做了什么关于library的变动后,最好都ldconfig一下,不然会出现一些意想不到的结果。不会花太多的时间,但是会省很多的事。

6,再有,诸如libdb-4.3.so文件头中是会含有库名相关的信息的(即含“libdb-4.3.so”,可用strings命令察看),因此仅通过修改文件名以冒充某已被识别的库(如libdb-4.8.so)是行不通的。为此可在编译库的Makefile中直接修改配置信息,指定特别的库名。

==================================================================================================================================================================

 

Ubuntu 下配置protobuf

 

http://blog.csdn.net/guoyilongedu/article/details/17093811

 

        最近想研究protobuf ,尝试了很多次都没有成功,我用的是ubuntu,在虚拟机下面的 ,protobuf 也用了很多版本但都没有成功。最终用的是2.5.0版本才成功,话不多说直接开始梳理一下配置的流程。  

 

  1. 首先得到  protobuf 相应的包文件 ,在终端上输入如下  
  2. wget http://protobuf.googlecode.com/files/protobuf-2.5.0.tar.gz  


下载完毕后进行解压

tar zxvf protobuf-2.5.0.tar.gz   

进入到解压目录

cd protobuf-2.5.0  

进行执行

./configure    


中间可能会出错,估计是G++没装好,因为安装的时候要进行编译

 

安装G++    

apt-get install g++  

另外最好把Vim、make 也装了,不然的后面的就很容易出问题,这些在其他教程上都没提到过,是个人的一点经验与大家分享一下

apt-get install vim  

 

apt-get install make  


./configure 成功之后,接下来 就如下几步

  1. make   
  2. make check  
  3. make install  

安装完成后在终端下执行

    vim ~/.profile  

 

 

打开配置文件,在该文件中添加

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib

然后保存退出,接下来执行

source ~/.profile  

是配置文件修改生效,最后执行

protoc --version  

 

查看protobuf版本以测试是否安装成功

接下来的操作 可以参照如下 链接 ,他们写得非常好 

运行,ubuntu下默认报错

 

protoc: error while loading shared libraries: libprotobuf.so.7: cannot open shared object file: No such file or directory

protoc: error while loading shared libraries: libprotoc.so.7: cannot open shared object file: No such file or directory

 

建一下硬链接

#cd /usr/lib

#sudo ln -s /usr/local/lib/libprotobuf.so.7 libprotobuf.so.7

#sudo ln -s /usr/local/lib/libprotoc.so.7 libprotoc.so.7

所以安装共享库后要注意共享库路径设置问题, 如下:

1) 如果共享库文件安装到了/lib或/usr/lib目录下, 那么需执行一下ldconfig命令

ldconfig命令的用途, 主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下, 搜索出可共享的动态链接库(格式如lib*.so*), 进而创建出动态装入程序(ld.so)所需的连接和缓存文件. 缓存文件默认为/etc/ld.so.cache, 此文件保存已排好序的动态链接库名字列表. 

2) 如果共享库文件安装到了/usr/local/lib(很多开源的共享库都会安装到该目录下)或其它"非/lib或/usr/lib"目录下, 那么在执行ldconfig命令前, 还要把新共享库目录加入到共享库配置文件/etc/ld.so.conf中, 如下:

# cat /etc/ld.so.conf
include ld.so.conf.d/*.conf
# echo "/usr/local/lib" >> /etc/ld.so.conf
# ldconfig

3) 如果共享库文件安装到了其它"非/lib或/usr/lib" 目录下,  但是又不想在/etc/ld.so.conf中加路径(或者是没有权限加路径). 那可以export一个全局变量LD_LIBRARY_PATH, 然后运行程序的时候就会去这个目录中找共享库. 

LD_LIBRARY_PATH的意思是告诉loader在哪些目录中可以找到共享库. 可以设置多个搜索目录, 这些目录之间用冒号分隔开. 比如安装了一个mysql到/usr/local/mysql目录下, 其中有一大堆库文件在/usr/local/mysql/lib下面, 则可以在.bashrc或.bash_profile或shell里加入以下语句即可:

export LD_LIBRARY_PATH=/usr/local/mysql/lib:$LD_LIBRARY_PATH    

一般来讲这只是一种临时的解决方案, 在没有权限或临时需要的时候使用.

 

http://hahaya.github.io/2013/08/12/use-protobuf-in-c-plus-plus.html

 

http://www.ccvita.com/507.html

 

创建一个proto 文件 比如 msg.proto

  1. package lm;     
  2. message helloworld     
  3. {     
  4.     required int32     id = 1;  // ID       
  5.     required string    str = 2;  // str      
  6.     optional int32     opt = 3;  //optional field     
  7. }  


将消息文件msg.proto映射成cpp文件

 

protoc -I=. --cpp_out=. msg.proto

 

可以看到生成了

msg.pb.h 和msg.pb.cc

http://blog.csdn.net/wallwind/article/details/11499643

 

对其代码做了一些纠正

write cpp 

  1. #include "msg.pb.h"  
  2. #include <iostream>  
  3. #include <fstream>  
  4. using namespace std;  
  5. using namespace lm;  
  6. int main(void)  
  7. {  
  8.     lm:helloworld msg1;  
  9.     msg1.set_id(101);  
  10.     msg1.set_str("hello");  
  11.     fstream output("./msg.pb",ios::out | ios::trunc | ios::binary);  
  12.     if( !msg1.SerializeToOstream(&output))  
  13.     {  
  14.          cerr << "Failed to write msg." << endl;     
  15.          return -1;    
  16.     }  
  17.     return 0;  
  18. }  


reader.cpp

 

  1. #include "msg.pb.h"  
  2. #include <iostream>  
  3. #include <fstream>  
  4. using namespace std;  
  5. using namespace lm;  
  6.   
  7. void listmsg(const lm::helloworld & msg)  
  8. {  
  9.     cout << msg.id() <<endl;  
  10.     cout << msg.str() <<endl;  
  11. }  
  12.   
  13. int main(void)  
  14. {  
  15.     lm:helloworld msg1;  
  16.     fstream input("./msg.pb", ios::in | ios::binary);     
  17.     if (!msg1.ParseFromIstream(&input)) {     
  18.             cerr << "Failed to parse address book." << endl;     
  19.             return -1;     
  20.     }     
  21.     listmsg(msg1);  
  22.     return 0;  
  23. }  

 

Makefile

  1. all: write reader    
  2.     
  3. clean:  
  4.     rm -f write reader msg.*.cc msg.*.h *.o  log    
  5.     
  6. proto_msg:  
  7.     protoc --cpp_out=. msg.proto    
  8.     
  9.     
  10. write: msg.pb.cc write.cpp  
  11.     g++  msg.pb.cc write.cpp -o write  `pkg-config --cflags --libs protobuf`    
  12.     
  13. reader: msg.pb.cc reader.cpp  
  14.     g++  msg.pb.cc reader.cpp -o reader  `pkg-config --cflags --libs protobuf`   

 

  1. 如果提示 make: Nothing to be done for 'all  

    

则执行make clean

如果从头开始执行的话

1.make clean

2.make proto_msg

3.make

package demo;  

message People

 { required string name = 1;   

required int32 id = 2;   

required string email = 3;  }


demo::People p;
p.set_name("guoyilong");
p.set_id(i);
p.set_email("guoyilong@163.com");
p.SerializeToString(&data);

当 i = 0 里 也就是 set_id(0)时 有一个很奇怪的现象,具体如下

data.length() 为32

strlen(data.c_str()); 则为 12 两个结果不一致

而i 不等于0 则不会出现这种情况 现在还是不解 记录一下
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值