Linux 下 C/C++ 静态库、动态库的区别

linux

Linux下的库必须以lib开头,用于系统识别

静态库 后缀 .a  每次被调用都生成一个副本


共享库(动态库) 后缀.so  只有一个副本


windows

静态库 后缀 .lib

动态库 后缀.dll


静态库的生成和使用


通常情况下,对函数库的链接是放在编译时期(compile time)完成的。所有相关的对象文件(object file)与牵涉到的函数库(library)被链接合成一个可执行文件(executable file)。程序在运行时,与函数库再无瓜葛,因为所有需要的函数已拷贝到自己门下。所以这些函数库被成为静态库(static libaray),通常文件名为“libxxx.a”的形式。


静态库必要的目标代码的是在对程序编译的时候被加入到程序中,而运行时不再需要.a的库了


生成静态库:

gcc -c try1.c  
gcc -c try2.c
ar cqs libtry.a try1.o try2.o(或 ar r libtry.a try1.o try2.o)

使用:

gcc -c main.c -o main.o  
gcc main.o -o name -L. -ltry (-l后面的名字就是我们上面生成的try库) 

动态库的生成和使用


动态链接库的特点与优势

首先让我们来看一下,把库函数推迟到程序运行时期载入的好处:

1. 可以实现进程之间的资源共享。

什么概念呢?就是说,某个程序的在运行中要调用某个动态链接库函数的时候,操作系统首先会查看所有正在运行的程序,看在内存里是否已有此库函数的拷贝了。如果有,则让其共享那一个拷贝;只有没有才链接载入。这样的模式虽然会带来一些“动态链接”额外的开销,却大大的节省了系统的内存资源。C的标准库就是动态链接库,也就是说系统中所有运行的程序共享着同一个C标准库的代码段。

2. 将一些程序升级变得简单。用户只需要升级动态链接库,而无需重新编译链接其他原有的代码就可以完成整个程序的升级。Windows 就是一个很好的例子。

3. 甚至可以真正坐到链接载入完全由程序员在程序代码中控制。

程序员在编写程序的时候,可以明确的指明什么时候或者什么情况下,链接载入哪个动态链接库函数。你可以有一个相当大的软件,但每次运行的时候,由于不同的操作需求,只有一小部分程序被载入内存。所有的函数本着“有需求才调入”的原则,于是大大节省了系统资源。比如现在的软件通常都能打开若干种不同类型的文件,这些读写操作通常都用动态链接库来实现。在一次运行当中,一般只有一种类型的文件将会被打开。所以直到程序知道文件的类型以后再载入相应的读写函数,而不是一开始就将所有的读写函数都载入,然后才发觉在整个程序中根本没有用到它们。



把一个源代码编译成.so


gcc -shared -o libtry.so try.c  




1.显式调用

不推荐


2.隐式调用

所谓隐式调用,就是调用的时候直接使用动态库中的函数,并不区别对待。

但是在隐式调用的时候必须要让程序能找到你所调用的函数的所属动态库。

我们先来建立一个感性认识:    

# more /etc/ld.so.conf 你会看到以下内容:   


/usr/kerberos/lib    

/usr/X11R6/lib    

/usr/lib/sane    

/usr/lib/qt-3.1/lib    

/usr/lib/mysql    

/usr/lib/qt2/lib    

/usr/local/lib    

/usr/local/BerkeleyDB.4.3/lib

 

ld.so.conf是系统对动态链接库进行查找的路径配置文件,也就是说该文件是系统链接工具/usr/bin/ld查找动态链接库的地图,所以,要达到我们的目的有以下几种方法:

a.将自己的动态链接库文件拷到以上路径的目录下

    # cp libwx.so.1 /usr/local/lib

b.将自己动态链接库文件的路径加入到该文件中

    # vi /etc/ld.so.conf然后加入自己的路径(pwd >>/etc/ld.so.conf)

    (千万不要将>>写成>,前者是添加,后者是覆盖)

c.把当前路径加入环境变量LD_LIBRARY_PATH,其实就是/usr/bin/ld的环境变量

    # export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH

    

编译的时候:    

# gcc -o try main.c libtry.so.1    

如果没有让/usr/bin/ld知道你的动态链接库在哪,编译的时候就要告诉它:    

# gcc -o try main.c /root/libtry.so.1 ( 或:# gcc -L/root/wx -o qqq main.c libmy.so.1)

-L指定动态链接库所在的目录,有时候用gcc还会碰到-I-l,他们分别指定头文件的目录和所链接的动态链接库

 

 

# ldd try    用来查看可执行文件qqq的动态链接库的依赖关系

        libjpeg.so.62 => /usr/lib/libjpeg.so.62 (0x00867000)

        libSDL-1.2.so.0 => /usr/lib/libSDL-1.2.so.0 (0x0638a000)

        libc.so.6 => /lib/tls/libc.so.6 (0x0046d000)

        libm.so.6 => /lib/tls/libm.so.6 (0x00598000)

        libdl.so.2 => /lib/libdl.so.2 (0x005bd000)

        libasound.so.2 => /lib/libasound.so.2 (0x062e1000)

        libX11.so.6 => /usr/X11R6/lib/libX11.so.6 (0x005d5000)

        libXext.so.6 => /usr/X11R6/lib/libXext.so.6 (0x0069e000)

        libpthread.so.0 => /lib/tls/libpthread.so.0 (0x006ae000)

        /lib/ld-linux.so.2 (0x00450000)






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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值