Linux静态库和动态库

来源:互联网 发布:真空上街 知乎 编辑:程序博客网 时间:2024/06/09 14:05

本文同步发布于我的个人网站:Linux静态库和动态库


Linux静态库和动态库

在windows平台和linux平台下都大量存在着库。本质上来说库是一种可执行的二进制代码(但不可以独立执行),可以被操作系统载入内存执行。

Linux下的库有两种:静态库和动态库(共享库)。

本文将介绍Linux下静态库和动态库的概念以及相应的创建与使用方法,文章内容为个人学习笔记,欢迎指正。

1 对比

类型 特点 静态库 在编译时被链接到程序中,作为可执行程序的一部分。在程序运行时不再依赖静态库,占用内存大 动态库 在可执行程序运行时载入内存。动态库已经在内存中不需要再次载入

2 静态库

2.1 概念

静态库指将所有相关的目标文件打包成为一个单独的文件,即静态库文件

静态库以.a结尾,链接器会将程序中使用到的函数的代码从库文件中拷贝到程序中。

由于每个使用静态库的应用程序都需要拷贝所有函数的代码,所以静态链接的文件会比较

在Unix系统中,静态库以一种称为存档(archive)的特殊文件格式存放在磁盘中。

存档文件是一组连接起来的可重定位目标文件的集合,有一个头部用来描述每个成员目标文件的大小和位置。

2.2 静态库的创建和使用

2.2.1 环境准备

创建根文件夹staticDemo:

[root@VM_120_243_centos ~]# mkdir staticDemo

创建子文件夹include,用于存放所有头文件:

[root@VM_120_243_centos ~]# cd staticDemo/[root@VM_120_243_centos staticDemo]# mkdir include

进入include文件夹,编写myLib.h文件,申明printInfo()函数:

[root@VM_120_243_centos staticDemo]# cd include/[root@VM_120_243_centos include]# vim myLib.h[root@VM_120_243_centos include]# cat myLib.h void printInfo();

进入上层目录,创建lib文件夹,用于存放所有库文件,再其中编写print.c文件,使用了myLib.h中的printInfo()函数:

[root@VM_120_243_centos include]# cd ..[root@VM_120_243_centos staticDemo]# mkdir lib[root@VM_120_243_centos staticDemo]# cd lib/[root@VM_120_243_centos lib]# vim print.c[root@VM_120_243_centos lib]# cat print.c #include <stdio.h>#include "myLib.h"void printInfo(){    printf("print from print.c file...\n");}

进入上层目录,编写main.c文件,用于测试:

[root@VM_120_243_centos lib]# cd ..[root@VM_120_243_centos staticDemo]# vim main.c[root@VM_120_243_centos staticDemo]# cat main.c #include <stdio.h>#include "myLib.h"int main(void){    printf("print from main.c file...\n");    printInfo();    return 0;}

准备工作到此完成,整个目录结构如下:

[root@VM_120_243_centos staticDemo]# tree.├── include│   └── myLib.h├── lib│   └── print.c└── main.c2 directories, 3 files
2.2.2 生成静态库

首先进入lib文件夹,生成print.o文件:

[root@VM_120_243_centos staticDemo]# cd lib/[root@VM_120_243_centos lib]# gcc -c print.c -I../include/ -o print.o[root@VM_120_243_centos lib]# lsprint.c  print.o

使用ar命令归档目标文件,得到静态库:

[root@VM_120_243_centos lib]# ar -crv libprint.a print.oa - print.o[root@VM_120_243_centos lib]# lslibprint.a  print.c  print.o

上述命令中crvar的命令选项:

  • c 如果需要生成新的库文件,不要警告

  • r 代替库中现有的文件或者插入新的文件

  • v 输出详细信息

使用-t参数可以查看静态库中包含的文件:

[root@VM_120_243_centos lib]# ar -t libprint.a print.o

注意:我们要生成的库的文件名必须形如libxxx.a,这样我们在链接这个库时,就可以用-lxxx

反过来讲,当我们告诉编译器-lxxx时,编译器就会在指定的目录中搜索libxxx.a或是libxxx.so

2.2.3 生成可执行文件

进入上层目录,链接库并生成可执行文件:

[root@VM_120_243_centos staticDemo]# gcc main.c -I./include/ -L./lib/ -lprint -o main

执行可执行文件:

[root@VM_120_243_centos staticDemo]# ./main print from main.c file...print from print.c file...

3 动态库

3.1 概念

动态库是一个目标模块,Linux系统以.so后缀表示,Windows以.dll后缀表示,使用动态库可以减小应用程序占用的空间和加载时间。

在运行时,可以加载到任意的存储器地址,并和一个再存储器中的程序链接起来,这个过程称为动态链接,是由一个叫做动态链接器的程序来执行的。

动态库是Linux系统最广泛的一种程序使用方式,工作原理是相同功能的代码可以被多个程序共同使用

在程序加载的时候,内核会检查程序使用到的动态库是否已经加载到内存:

  • 如果没有被加载到内存,则从系统库路径搜索并且加载到相关的动态库。

  • 如果动态库已经被加载到内存,程序可以直接使用而无需加载。

应用程序在自身加载时运行过程中动态链接和加载动态库。

3.2 动态库的创建和使用

3.2.1 环境准备

创建根文件夹dynamicDemo:

[root@VM_120_243_centos ~]# mkdir dynamicDemo

创建子文件夹include,用于存放所有头文件:

[root@VM_120_243_centos ~]# cd staticDemo/[root@VM_120_243_centos dynamicDemo]# mkdir include

进入include文件夹,编写myLib.h文件,申明printInfo()函数:

[root@VM_120_243_centos dynamicDemo]# cd include/[root@VM_120_243_centos include]# vim myLib.h[root@VM_120_243_centos include]# cat myLib.h void printInfo();

进入上层目录,创建lib文件夹,用于存放所有库文件,再其中编写print.c文件,使用了myLib.h中的printInfo()函数:

[root@VM_120_243_centos include]# cd ..[root@VM_120_243_centos dynamicDemo]# mkdir lib[root@VM_120_243_centos dynamicDemo]# cd lib/[root@VM_120_243_centos lib]# vim print.c[root@VM_120_243_centos lib]# cat print.c #include <stdio.h>#include "myLib.h"void printInfo(){    printf("print from print.c file...\n");}

进入上层目录,编写main.c文件,用于测试:

[root@VM_120_243_centos lib]# cd ..[root@VM_120_243_centos dynamicDemo]# vim main.c[root@VM_120_243_centos dynamicDemo]# cat main.c #include <stdio.h>#include "myLib.h"int main(void){    printf("print from main.c file...\n");    printInfo();    return 0;}

准备工作到此完成,整个目录结构如下:

[root@VM_120_243_centos dynamicDemo]# tree.├── include│   └── myLib.h├── lib│   └── print.c└── main.c2 directories, 3 files
3.2.2 生成动态库

进入lib文件夹,给gcc加入-fPIC参数,生成print.o文件:

[root@VM_120_243_centos dynamicDemo]# cd lib/[root@VM_120_243_centos lib]# gcc -c print.c -fPIC -I../include/ -o print.o[root@VM_120_243_centos lib]# lsprint.c  print.o

使用gcc的-shared参数生成动态库libprint.so:

[root@VM_120_243_centos lib]# gcc -shared print.o -o libprint.so[root@VM_120_243_centos lib]# lslibprint.so  print.c  print.o
3.2.3 生成可执行文件

进入dynamicDemo目录,链接库并生成可执行文件:

[root@VM_120_243_centos dynamicDemo]# gcc main.c -I./include/ -L./lib/ -lprint -o main

注意:如果同一目录下同时存在同名的动态库和静态库,比如libprint.solibprint.a都在当前路径下,则gcc会优先链接动态库

执行可执行文件:

[root@VM_120_243_centos dynamicDemo]# ./main ./main: error while loading shared libraries: libprint.so: cannot open shared object file: No such file or directory

系统报错找不到libprint.so,原来Linux是通过/etc/ld.so.cache文件搜寻要链接的动态库的。而/etc/ld.so.cacheldconfig程序读取/etc/ld.so.conf文件生成的。(注意,/etc/ld.so.conf中并不必包含/lib/usr/libldconfig程序会自动搜索这两个目录)

这里我们指定环境变量来找到动态库路径,在下一节中会有详细介绍:

[root@VM_120_243_centos dynamicDemo]# LD_LIBRARY_PATH=./lib/ ./mainprint from main.c file...print from print.c file...

3.3 设置动态库的搜索路径

3.3.1 LD_LIBRARY_PATH

LD_LIBRARY_PATH是Linux环境变量名,该环境变量主要用于在查找动态库时搜索默认路径之外的其他路径。

通常,Linux搜索动态库的默认路径是/lib/usr/libLD_LIBRARY_PATH路径优先级大于系统默认路径。

[root@VM_120_243_centos dynamicDemo]# gcc main.c -I./include/ -L./lib/ -lprint -o main[root@VM_120_243_centos dynamicDemo]# LD_LIBRARY_PATH=./lib/ ./mainprint from main.c file...print from print.c file...
3.3.2 rpath

gcc的-R或者-rpath选项在编译链接时指定动态库路径,并且将动态库路径保存到可执行文件中。

运行可执行文件时直接到该路径下查找动态库,而不依赖于默认路径或者环境变量。

[root@VM_120_243_centos dynamicDemo]# gcc main.c -I./include/ -L./lib/ -lprint -Wl,-rpath=./lib -o main[root@VM_120_243_centos dynamicDemo]# ./main print from main.c file...print from print.c file...

采用这种方式,每个程序可以设定独立的动态库位置,而不会像LD_LIBRARY_PATH环境变量那样影响其他程序。

3.3.3 ldconfig

ldconfig是一个动态链接库管理命令。该命令的用途是在默认路径(/lib/usr/lib)以及动态库配置文件/etc/ld.so.conf内所包含的目录下,搜索动态库,然后为动态载入程序(ld.so)创建动态库列表的缓存文件(/etc/ld.so.cache)

3.3.3.1 加入标准路径

这里以放入/usr.lib文件夹为例:

[root@VM_120_243_centos dynamicDemo]# cp ./lib/libprint.so /usr/lib/

更新缓存:

[root@VM_120_243_centos dynamicDemo]# ldconfig

查看动态库:

[root@VM_120_243_centos dynamicDemo]# ldconfig -p | grep libprint.so    libprint.so (libc6,x86-64) => /lib/libprint.so

编译执行:

[root@VM_120_243_centos dynamicDemo]# gcc main.c -I ./include/ -L ./lib/ -lprint -o main[root@VM_120_243_centos dynamicDemo]# ./main print from main.c file...print from print.c file...
3.3.3.2 加入配置文件

首先查看/etc/ld.so.conf文件:

[root@VM_120_243_centos dynamicDemo]# cat /etc/ld.so.confinclude ld.so.conf.d/*.conf

可以看到其包含了ld.so.conf.d目录,因此自定义配置文件libprint.conf加入到/etc/ld.so.conf.d文件夹中,配置文件中包含该程序的动态库路径:

[root@VM_120_243_centos dynamicDemo]# pwd/root/dynamicDemo[root@VM_120_243_centos dynamicDemo]# vim /etc/ld.so.conf.d/libprint.conf[root@VM_120_243_centos dynamicDemo]# cat /etc/ld.so.conf.d/libprint.conf/root/dynamicDemo/lib

更新缓存:

[root@VM_120_243_centos dynamicDemo]# ldconfig

查看动态库:

[root@VM_120_243_centos dynamicDemo]# ldconfig -p | grep libprint.so    libprint.so (libc6,x86-64) => /root/dynamicDemo/lib/libprint.so

编译执行:

[root@VM_120_243_centos dynamicDemo]# gcc main.c -I ./include/ -L ./lib/ -lprint -o main[root@VM_120_243_centos dynamicDemo]# ./main print from main.c file...print from print.c file...

参考文献

[1] JollyWing. Linux静态库生成指南[EB/OL]. http://www.cnblogs.com/jiqingwu/p/4325382.html.

[2] JollyWing. Linux动态库生成与使用指南[EB/OL]. http://www.cnblogs.com/jiqingwu/p/linux_dynamic_lib_create.html.

原创粉丝点击