• ldconfig 与 ldd 动态链接库管理

    背景

    我想你肯定遇到过这种情况,在执行某个新安装的启动命令时,报出类似于下面的错误:

    $  ./sbin/nginx -t

    ./sbin/nginx: error while loading shared libraries: libluajit-5.1.so.2: cannot open shared object file: No such file or directory

    看似很明了,缺少某个动态链接库,但其实处理起来很头痛。 有经验的运维可能会想到。 哦,可能是我没有安装luagit导致的,于是执行安装luagit以后再重试:

    $ yum install luagit

    $ ./sbin/nginx -t

    ./sbin/nginx: error while loading shared libraries: libpcre.so.0: cannot open shared object file: No such file or directory

    又来一个,那么我根据上次的方法,执行安装pcre,然后再重试:

    $ yum install pcre

    $ ./sbin/nginx -t

    ./sbin/nginx: error while loading shared libraries: libpcre.so.0: cannot open shared object file: No such file or directory

    哎呀,貌似这招不管用了。于是只好在网上搜了一下:

    $ link /usr/lib64/libpcre.so.1 /lib64/libpcre.so.0

    $ ./sbin/nginx -t

    nginx: the configuration file /etc/nginx/conf/nginx.conf syntax is ok

    nginx: configuration file /etc/nginx/conf/nginx.conf test is successful

    这下貌似才好了,可是这样感觉好被动啊,我不能主动的去发现启动nginx到底需要哪些动态链接库吗 ?当然可以:

    $ ldd $(which /usr/local/nginx/sbin/nginx)

    %e5%b1%8f%e5%b9%95%e5%bf%ab%e7%85%a7-2017-01-04-%e4%b8%8b%e5%8d%887-24-57

    如上,使用ldd命令就可以显示出我们启动nginx 命令需要有哪些依赖库,而目前还有哪些依赖库没有找到。比如上图中的libpcre.so.0 这个库还没有找到。

    1. 那么问题来了,对于Linux系统来说,它一般默认从哪里寻找我需要的这些库呢?
    2. 比如当我用yum或者源码安装pcre后,该怎样让pcre的lib库加入系统可以运用的大家庭呢?

    我们要从ldconfig 和 ldd 这两个命令说起:

    ldconfig

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

    到此我们的问题1得到解决了,Linux一般从/lib  和 /usr/lib 和 /etc/ld.so.conf内所列的目录下寻找我们需要的动态库。

    ldconfig的使用:

    • 往/lib和/usr/lib里面加东西,是不用修改/etc/ld.so.conf的,但是完了之后要调一下ldconfig,不然这个library会找不到,ldconfig这个命令会更新/etc/ld.so.cache,系统是利用这个cache来寻找的
    • 想往上面两个目录以外加东西的时候,一定要修改/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才能在程序运行时被找到。
    • 如果想在这两个目录以外放lib,但是又不想在/etc/ld.so.conf中加东西(或者是没有权限加东西)。那也可以,就是export一个全局变量LD_LIBRARY_PATH,然后运行程序的时候就会去这个目录中找library。一般来讲这只是一种临时的解决方案,在没有权限或临时需要的时候使用。
    • ldconfig做的这些东西都与运行程序时有关,跟编译时一点关系都没有。编译的时候还是该加-L就得加,不要混淆了。 总之,就是不管做了什么关于library的变动后,最好都ldconfig一下,不然会出现一些意想不到的结果。不会花太多的时间,但是会省很多的事。

    参考:

    http://man.linuxde.net/ldconfig