前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >linux下的 lib文件的学习思考

linux下的 lib文件的学习思考

作者头像
孙杰
发布2019-10-29 14:25:01
2.5K0
发布2019-10-29 14:25:01
举报
文章被收录于专栏:云技术+云运维云技术+云运维

说到这个LIB文件,先从一个小故障说起。

某日开发说,一台测试用虚机可以PING通SSH不能连了。运维同学就赶紧去查,SSHD_CONFIG配置文件都正确啊,一点错误都没有,那为什么呢?

测试下,不管连自己还是其他机,都是报错

wKiom1j1tNiwG6XzAABTYHuYQMc027.jpg
wKiom1j1tNiwG6XzAABTYHuYQMc027.jpg

这里注意看,提示你有个libcom_err.so.2共享库文件找不到。

询问开发,才了解他们测试一个软件,意外删除了某个库文件。

那么在正常的相同虚机的机器查看下,再和出错的虚机比对下,发现少了2个库文件

wKioL1j1tdfzehqVAAB-bnDvJSA177.jpg-wh_50
wKioL1j1tdfzehqVAAB-bnDvJSA177.jpg-wh_50

挂载系统光盘或从正常的虚机上把这个两个文件拷贝过来,放到lib64下就可以了

再试正常了

wKioL1j1thLBu3avAABQJqKDyuc847.jpg-wh_50
wKioL1j1thLBu3avAABQJqKDyuc847.jpg-wh_50

这个小故障很容易解决,那么你怎么理解linux中的库文件呢?学习下也不误工作。

Linux下的库文件分为共享库和静态库两大类,它们两者的差别仅在程序执行时所需的代码是在运行时动态加载的,还是在编译时静态加载的。 

Linux的库一般在/lib 或/usr/lib 目 录下,如果是64位的系统则会有lib64目录。lib是库(Library)的英文缩写,它主要存放系统的链接库文件,没有该目录则系统就无法正常运行。/lib目录中存储着程序运行时使用的共享库。通过共享库,许多程序可以重复使用相同的代码,并且这些库可以存储在一个公共的位置上,因此能减小运行程序的大小。这个目录包含程序在链接时使用的 各种库。

库的知识

1.库的命名

库的命名比较简单,第一个特点是所有的库以lib开头,GCC命令在在-l选项所指定的文件名前会自动加入lib。

第二个特点文件名以.a结尾的库是静态库。

第三个特点文件名是.so的库为共享库(共享库是在运行的时候动态加载的 )。默认情况下,GCC在链接时优先使用共享库,只有当共享库不存在时才考虑使用静态库。 

2、库的编号

库的编号格式如下:  library_name .major.num .minor_.min .pathch_num 例如,笔者Red Hat Linux 9.0的GUN数据库是libgdbm.so.0.0.2,详细表述如下: ◆library_name是libc.so(标准C库); ◆major_num是2(主版本号); ◆minor_.min是0(次版本号); ◆pathch_num是0(补丁级别号又称发行号)。

3、库的操作命令

Linux库操作可以使用命令完成,目前常用的命令是ldd和ldconfig。 ldd 是Library Dependency Display缩写,它的作用是显示一个可执行程序必须使用的共享库 。 (1)命令格式 ldd [选项] 文件名 (2)主要参数 -d 执行重定位并报告丢失的函数。 -r 执行对函数和数据对象的重定位,并报告丢失的函数和数据对象。

(3)应用举例

比如查询Perl语言有哪些共享库,则可以首先使用find命令查询这个程序的绝对路径,然后使用ldd命令: #find -name perl ldd /usr/bin/perl

$ ldd test 执行test,可以看到它是如何调用动态库中的函数的。

2.ldconfig ldconfig 命令的作用是决定位于目录/usr/lib和/lib下的共享库所需的运行链接。这些链接保存在的Libs保存在/et/ld.so.conf文件中。搜 索出可共享的动态链接库(格式如前介绍,lib*.so*),进而创建出动态装入程序(ld.so)所需的链接和缓存文件。缓存文件默认为/etc /ld.so.cache,此文件保存已排好序的动态链接库名字列表。 (1)命令格式 ldconfig [选项] [libs] (2)主要选项  -v或--verbose ldconfig将显示正在扫描的目录、搜索到的动态链接库,以及它所创建的连接的名字。  -f CONF 指定动态链接库的配置文件为CONF,系统默认为/etc/ld.so.conf。  -C CACHE 指定生成的缓存文件为CACHE,系统默认的是/etc/ld.so.cache,文件存放已排好序的可共享的动态链接库的列表。  -p或--print-cache 让ldconfig打印出当前缓存文件所保存的所有共享库的名字。  -r ROOT 改变应用程序的根目录为ROOT。  -n ldconfig仅扫描命令行指定的目录,不扫描默认目录(/lib、/usr/lib),也不扫描配置文件/etc/ld.so.conf所列的目录。  运行没有选项的ldconfig命令时,用于更新高速缓冲文件。这个命令主要用于高速缓冲DNS服务器(Caching DNS Server)。高速缓冲DNS服务器的原理是提供查询的历史记录,并且利用这些记录来提高查询的效率。  当某个查询是第一次被发送到高速缓冲DNS服务器时,高速缓冲DNS服务器就将此查询的整个过程记录下来,在一定的时期内用它来回答所有相同的查询,从而减少整个DNS系统的负担并且提高查询速度。  (3)应用实例  如果用户想知道系统中有哪些动态链接库,或者想知道系统中有没有某个动态链接库时,可用-p选项让ldconfig输出缓存文件中的动态链接库列表,从而查询得到。 例如:

ldconfig -p 998 libs found in cache `/etc/ld.so.cache' libzvt.so.2 (libc6) => /usr/lib/libzvt.so.2 libzvt.so (libc6) => /usr/lib/libzvt.so ……

补充:

静态链接库*.a的编译和使用 创建.a库文件和.o库文件: [yufei@localhost perl_c2]$ pwd /home/yufei/perl_c2 [yufei@localhost perl_c2]$ cat mylib.c #include <stdio.h> #include <string.h> void hello(){        printf("success call from perl to c library\n"); } [yufei@localhost perl_c2]$ cat mylib.h extern void hello(); [yufei@localhost perl_c2]$ gcc -c mylib.c [yufei@localhost perl_c2]$ dir mylib.c  mylib.h  mylib.o [yufei@localhost perl_c2]$ ar -r mylib.a mylib.o ar: 正在创建 mylib.a [yufei@localhost perl_c2]$ dir mylib.a  mylib.c mylib.h  mylib.o

*.a的使用方法

最简单的是直接把.a当成一个普通源代码编译进来.  gcc   main.cpp   ./lib/libInfo.a   -o   exec

动态链接库*.so的编译与使用- - 动态库*.so在linux下用c和c++编程时经常会碰到,这里做个笔记,也为其它正为动态库链接库而苦恼的兄弟们提供一点帮助。 1、动态库的编译 下面通过一个例子来介绍如何生成一个动态库。这里有一个头文件:so_test.h,三个.c文件:test_a.c、test_b.c、test_c.c,我们将这几个文件编译成一个动态库:libtest.so。 so_test.h: #include <stdio.h> #include <stdlib.h> void test_a(); void test_b(); void test_c(); test_a.c: #include "so_test.h" void test_a() {     printf("this is intest_a...\n"); } test_b.c: #include "so_test.h" void test_b() {     printf("this is intest_b...\n"); } test_c.c: #include "so_test.h" void test_c() {     printf("this is intest_c...\n"); } 将这几个文件编译成一个动态库:libtest.so $ gcc test_a.c test_b.c test_c.c -fPIC -shared -o libtest.so 2、动态库的链接 在1、中,我们已经成功生成了一个自己的动态链接库libtest.so,下面我们通过一个程序来调用这个库里的函数。程序的源文件为:test.c。 test.c: #include "so_test.h" int main() {     test_a();     test_b();     test_c();     return 0; } l       将test.c与动态库libtest.so链接生成执行文件test: $ gcc test.c -L. -l test -o test l       测试是否动态连接,如果列出libtest.so,那么应该是连接正常了 $ ldd test l       执行test,可以看到它是如何调用动态库中的函数的。

总结: 1、共享库特别适合多个程序共享代码,升级程序部分功能模块,实现程序“插件”功能的情况; 而静态库是一劳永逸,编译后不需要带一堆库文件跑,而且不管放置到哪里都可正常运行。 2、当搜索的库文件目录下同时存在该库的静态版本和共享版本时,链接器优先使用共享版本.so,此时你可以使用-static链接选项指定链接静态版本.a。 3、动态库可以导出两个特殊的函数:_init和_fini,前者在动态库被加载后调用,后者在动态库被卸载前调用, 我们可以使用这两个函数做些特别的工作。需要注意的是:在定义这两个函数后编译时,需要使用 -nostartfiles选项,否则编译器报重复定义错误。 4、ldd命令用来查看程序所依赖的共享库,同时也方便我们判断共享库是否被找到;    nm命令查看obj文件(.so也是一个obj)中的标识(函数、变量)。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2017-04-18 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档