• Linux:动静态库


    目录

    一、软硬链接

    1、软链接

    2、硬链接

    二、动态库和静态库

    编写一个库

    ①静态库

    使用静态库的方法

    ②动态库

    使用动态库的方法

    库存在的意义


    一、软硬链接

    软硬链接的本质区别就是:有无独立的inode

    软链接有独立的inode,也就意味着软链接是一个独立的文件

    硬链接没有独立的inode,即硬链接并不是一个独立的文件

    ln -s是软链接的语法,不加-s就是硬链接

    上图将soft.txt与testlink1软链接,将hard.txt与testlink2硬链接,通过ls -li(-i表示显示对应的inode值),可以看出进行软链接的soft.txt与testlink1的inode不同,而进行硬链接的hard.txt与testlink2的inode却是相同的,即表示硬链接没有独立的inode

    1、软链接

    软链接相当于Windows下的快捷方式,如下所示:

    我们有一个文件夹test

    test里还有个文件夹hello,hello里有个文件test.c,test.c经过gcc形成可执行文件test.exe,test.c文件内容如下:

    如果在当前路径下,想运行这个test.exe这个可执行文件:

    即使通过相对路径的方式运行,也是比较麻烦的,所以这时使用软链接:

    将该路径下的test.exe与test/hello/下的test.exe建立软链接,这时如果想执行刚刚的可执行文件,只需运行test.exe即可:

    所以软链接相当于Windows下的快捷方式

    可以理解为:软链接的文件内容,是指向的文件对应的路径

    unlink删掉软链接


    2、硬链接

    由于硬链接没有独立的inode,所以硬链接不是独立的文件,所以创建硬链接并不是真正的创建新文件,而是在指定的目录下,建立了文件名和指定inode的映射关系,也就是起别名

    在ls -li时,我们可以发现这样一个属性:

    这样的数字表示的就是硬链接数

    例如我们上面将hard.txt与testlink2硬链接了,所以他们共同的这个硬链接数属性就是2:

    而如果删除其中一个:

    这时hard.txt的硬链接数就变为了1

    所以当我们删除一个文件时,并不是删除文件的inode,而是将inode中包含的引用计数count即硬链接数--,直到引用计数为0,这个文件才真正的删除了


    下面看下一个问题,分别创建一个目录dir和一个普通文件test.c:

    为什么dir的硬链接数是2,而普通文件的硬链接数是1

    其实很简单,当我们创建普通文件test.c时,创建出来后,test.c这个文件名映射到自己的inode,所以硬链接数为1

    而目录dir,创建出来后,自己的目录名dir映射到自己的inode,这是其中一个;自己目录内部有一个  .  ,也与inode有对应的映射关系,这时第二个,因此目录的硬链接数默认为2

    这也就是我们平时运行可执行文件时是./test.exe,前面的.也就是表示是在当前路径下


    而如果我们在dir目录下,再创建一个dir1目录:

    这时dir的硬链接数变为了3,这又是什么原因呢

    其实也很简单,我们分别进入dir和dir1,都进行ls -lia,列出详细信息

    可以发现dir中的 . 与dir1中的 .. 的inode都是790425,所以我们就清楚这里的第三个硬链接数就是来自dir1中的 .. 

    因此也就能解释,为什么cd ..就是返回上一级路径,原因就是这里的..与上一级路径的inode是相同的,也就是当前路径的..就是上一级路径的别名,所以cd ..就是返回上一级路径

    .和..就是下图所表示的关系:

    因此如果我们继续在dir中创建目录,硬链接数也就会增加:

    原本是3,再mkdir两个就变为5,经过上面讲解,就可以知道新增的2个就是dir2和dir3中的 .. ,所以通过这一点,我们以后观察目录的硬链接数数是多少,只需要-2(减去的是dir与dir中的.这两个对应关系),就是里面包含的目录数

    所以上面的dir硬链接数是5,-2即包含的目录有三个,分别是dir1/2/3:


    二、动态库和静态库

    关于动静态库,我们前面的博客也提到过,Linux中,静态库是.a结尾,动态库是.so结尾 

    下面要讲到的动静态库的制作和使用,都是为了后续更好的使用别人的库做准备

    编写一个库

    首先明确一点,库中不能有main函数,因为库是给别人用的,如果有main函数,别人也有main函数,就重复定义了

    ①静态库

    首先有一个mklib的目录(制作库),里面有四个文件,分别是:

    myadd.h、myadd.c、myshow.h、myshow.c

    myadd.h

    myadd.c

    myshow.h

    myshow.c

    然后创建一个目录uselib(使用库),

    里面有main.c,用于使用myshow和myadd

    下面可以初步使用一下,首先将mkdir中的myadd.c和myshow.c变为.o文件:

    然后将所有.o和.h文件都cp到uselib中,再将main.c也变为.o文件:

    此时uselib中就有全部的.c和.h文件,这时生成可执行exe文件并运行:

    是能够运行的


    但是这种方式.o文件太多了,拷贝给别人使用时不方便,且编译时这些.o文件都需要打上去,万一不留神少打一个.o文件,就会出问题

    所以我们可以把所有的.o打包,形成一个静态库

    语法是:ar -rc [库名] .o文件 

    ar是归档工具,是归档文件的缩写

    -rc中r表示替换,c表示创建

    库名:前缀必须是lib,后缀必须是.a

    这里的libtest.a就是创建出来的静态库

    为了方便,创建Makefile文件:

    所以我们就可以make clean清除.o文件与静态库,make生成.o与静态库:

    而发布库时,需要include目录和lib目录:

    include目录,包含库的所有头文件

    lib目录,包含对应的库文件

    所以此时在我们的Makefile文件中新增内容:

    此时make test,就会生成test的目录,里面有include和lib目录:


    至此有了test目录,我们将该目录拷贝到uselib中去使用:

    test就是别人的库,而main.c是我准备使用别人的库的main函数

    使用静态库的方法

    第一种方法:拷贝到库目录下:

    头文件gcc的默认搜索路径是:/usr/include

    库文件的默认搜索路径是: /lib64

    所以将头文件与库文件拷贝到库目录下,编译时就能够找到它们了

    将所有.h拷贝到/usr/include中:

    这时/usr/include 下就有了myadd.h和myshow.h了

    再将libtest.a拷贝到/usr/lib中:

    这时/lib64中就有了libtest.a了

    而自己写的库是属于第三方库

    我们刚刚所写的库名字叫做libtest.a,所以我们gcc时必须表明需要链接的库名,前面加上-l,且将前缀lib和后缀.a都去掉就是库名,即test

    生成了一个可执行文件a.out,运行a.out:

    至此运行成功,成功使用了静态库libtest.a

    我们刚刚的拷贝库到系统的默认路径下,就叫做库的安装

    这第一种方法拷贝到库目录下,是不建议的,因为我们自己写的方法并不具有可靠性


    第二种方法:直接使用静态库

    如果我们直接gcc main.c,会报错:

    告诉我们找不到myadd.h,因为默认在库中搜索,搜索不到再到当前路径搜索,而当前路径是uselib,里面并没有.h文件,.h文件在./test/include中,所以-I表示指定让头文件搜索的路径

    这时报错找不到show和add函数,所以-L表示库文件的搜索路径,后面-l跟上特定路径下的库名

    这时成功生成a.out可执行文件,./a.out运行成功

    总结gcc main.c -I ./test/include -L ./test/lib -ltest

    -I(大写)是头文件的搜索路径,-L是库文件的搜索路径-l(小写)加上特定路径下的库名


    ②动态库

    动态库的四个函数内容不变,.h和.c也在mklib中,和上面静态库一样

    形成的动态库的.o文件必须要加-fPIC选项,.o文件名称加个_d,与静态库保持区分:

    而形成动态库,在gcc时需要加上-shared:

    -o后面的libtest.so,lib是前缀,.so是后缀,中间的test是库名

    此时就生成了libtest.so的动态库

    此时了解如何生成的动态库后,就可以把这个过程也编入到Makefile中:

    上面的.PHONY:all,是为了一次同时生成静态库和动态库

    此时执行make:

    可以发现同时生成了静态库和动态库

    再执行make test,就会生成test的目录,里面有include和lib目录:


    同样我们将该目录拷贝到uselib中去使用:

    此时uselib中内容有:

    此时动静态库都存在,编译器默认使用动态库:

    使用了gcc main.c -I ./test/include -L ./test/lib -ltest,运行a.out时,提醒无法使用动态库,证明了上述结论,使用ldd a.out查看同样可以发现使用的是动态库:

    而动静态库都存在时,我们如果想强行使用静态库,需要加上-static选项:

    此时就能够执行a.out了

    所以站在使用者的角度,得出以下结论:

    如果我们只有静态库,gcc只能针对该库进行静态链接

    如果动静态库同时存在,默认使用动态库

    如果动静态库同时存在,我们加上-static选项可以强行使用静态库

    -static的作用:改变默认优先使用动态库的原则,变为直接使用静态库的方案


    使用动态库的方法

    动态库同样可以像静态库那样,直接拷贝到库目录下,但是我们自己写的库不具有可靠性,所以并不建议这样做,所以这里就不举例这种方式了

    第一种方法:添加库所在的位置到LD_LIBRARY_PATH中(临时方案)

    LD_LIBRARY_PATH:库加载的搜索路径

    系统在搜索库路径时,会在系统的lib64路径下搜素,找到动态库就会使用,如果没找到就终止,如果系统的lib64路径没找到,而LD_LIBRARY_PATH路径也设置了,那就可以在这个路径下搜索

    原始的LD_LIBRARY_PATH有下面内容:

    我们自己的动态库在这个路径:/home/fcy/lesson2/uselib/test/lib

    所以执行:

    export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/fcy/lesson2/uselib/test/lib

    将该路径添加到LD_LIBRARY_PATH中,LD_LIBRARY_PATH=$LD_LIBRARY_PATH:是为了添加时不覆盖之前的内容,用冒号隔开

    执行完后,再查看LD_LIBRARY_PATH,就发现刚刚动态库所在的路径被添加到LD_LIBRARY_PATH里面了

    这时运行a.out就能成功运行了:

    ldd查看a.out:

    就会发现,可以查到我们自己的动态库libtest.so

    但是这种方法,如果你把你的Xshell软件关了,重新打开一遍,刚刚做的工作就又失效了,所以这种方法只能作为临时方案


    第二种方法:添加库所在的位置到/etc/ld.so.conf.d中(永久方案)

    系统里有一个路径/etc/ld.so.conf.d,里面保存的是可以自定义配置搜索库路径的永久解决方案:

    操作方法很简单,我们先在/etc/ld.so.conf.d里新创建一个普通文件tmp.conf,以.conf结尾(sudo创建)

    然后将我们刚刚动态库所在的路径,sudo方式vim打开并粘贴进去:

    sudo vim /etc/ld.so.conf.d/tmp.conf

    sudo ldconfig,就是让配置文件生效,更新一下

    然后运行a.out就能成功运行

    ldd a.out也能查看能够找到动态库

     


    第三种方法:软链接方案

    在/lib64目录下,创建一个软链接,链接到我们的动态库路径下

    这时执行a.out,就能成功执行了:

    ldd a.out查看也能找到动态库:

    找到的是lib64/libtest.so,而lib64/libtest.so是软链接,所以找到了动态库

    不需要时再将这个软链接删掉即可,sudo unlink /lib64/libtest.so


    库存在的意义

    对于这些库,自然有他们存在的意义

    对于使用者来说:

    有了库,可以大大减少我们开发的周期,从而提高我们软件的质量

    对于开发库的人员角度来说:

    第一、使用者使用简单

    第二、为了代码安全


  • 相关阅读:
    3d360全景家具展厅制作的优势盘点
    《复盘+》把经验转化为能力
    搞透 IOC,Spring IOC 看这篇就够了!
    Pycharm项目使用pyinstalle打包过程中问题及解决方案
    Allegro基本规则设置指导书之Analysis Modes
    IPV4优先于IPV6设置
    操作系统——环境变量
    linux中常见服务端安装
    Vue项目实战——【基于 Vue3.x + NodeJS】实现的课程表排课系统三(duration-title)
    融云 CDN 播放器 2.0 版本正式上线
  • 原文地址:https://blog.csdn.net/m0_64411530/article/details/134468725