Unix的静态库与动态库

前言

为了调用者使用的方便,一般并不会直接提供对应的.c文件或者.o文件,而是根据具体的功能模块,将对应的多个.o文件打包一个/多个库文件,给调用者提供库文件和头文件即可,系统默认提供了标准库还有一些其他的库文件,用户也可以自定义函数库,那么根据库连接的方式不同主要将函数库分为共享库(动态库)和静态库两种。

静态库

目录

1.概述
2.创建静态库以及使用静态库

概述

静态库:也叫做归档文件,以.a结尾。
使用静态库的时候,编译器在编译过程中直接将代码嵌入到目标文件中,所以一旦完成编译,那么静态库可以不需要了。
静态库的优点:不需要跳转,执行效率相对比较高。
静态库的缺点:目标文件相对比较大,并且后期的修改以及维护不方便。
静态库文件命名规范:lib(库名).a 比如:libadd.a
库文件名和库名是不同的概念,库名没前缀和后缀

创建静态库

1.创建头文件(add.h)

#ifndef __HH_ADD_H__
#define __HH_ADD_H__
int add(int,int);
#endif

2.创建源程序代码文件(add.c)

#include "add.h"
int add(int num1,int num2){
    return num1+num2;
}

3.编译源程序 生成目标文件 (add.o)

gcc -c add.c

4.使用ar打包工具生成静态库

ar -r 静态库文件 目标文件
ar -r libadd.a add.o

5.编写测试文件(main.c)

#include <stdio.h>
#include "add.h"

int main(){
    printf("1+1=%d",add(1,1));
    return 0;
}

6.使用静态库文件
第一种使用方式:直接链接

gcc 目标文件(源代码文件) 静态库文件
gcc main.c libadd.a
lurongshuang@ubuntu:~/work/work1020$ gcc main.c libadd.a
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$ 

第二种使用方式:使用编译选项进行链接 (这种方式使用较多)

gcc/cc 目标文件(源代码文件) -l 库名 -L 库文件所在的路径
gcc main.c -l add -L .
lurongshuang@ubuntu:~/work/work1020$ rm a.out 
lurongshuang@ubuntu:~/work/work1020$ gcc main.c -l add -L.
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$

第三种使用方式:配置环境变量 LIBRARY_PATH

export LIBRARY_PATH=路径
gcc 目标文件(源代码文件) -l 库名
lurongshuang@ubuntu:~/work/work1020$ gcc main.c -l add
/usr/bin/ld: 找不到 -ladd
collect2: error: ld returned 1 exit status
lurongshuang@ubuntu:~/work/work1020$ export LIBRARY_PATH=$LIBRARY_PATH:.
lurongshuang@ubuntu:~/work/work1020$ gcc main.c -l add
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$

动态库

目录

1.概述
2.创建动态库以及使用动态库

概述

动态库也叫做共享对象库、共享库等,以.so结尾。
使用动态库时,编译器并不直接在编译的时候将代码嵌入到目标文件中,而是等到运行时调用相应的函数,才加载代码。
动态库的优点:相比静态库 目标文件小,后期的修改维护方便。
动态库的缺点:因为运行时需要跳转,效率会低,并且不能脱离共享库文件。
动态态库文件命名规范:lib(库名).so 比如:libadd.so
库文件名和库名是不同的概念,库名没前缀和后缀

创建动态库以及使用静态库

1.创建头文件(add.h)

#ifndef __HH_ADD_H__
#define __HH_ADD_H__
int add(int,int);
#endif

2.创建源程序代码文件(add.c)

#include "add.h"
int add(int num1,int num2){
    return num1+num2;
}

3.编译源程序 生成目标文件 (add.o)

gcc -c -fpic 源代码文件
gcc -c -fpic add.c
lurongshuang@ubuntu:~/work/work1020$ gcc -c -fpic add.c
lurongshuang@ubuntu:~/work/work1020$ ls
add.c  add.h  add.o  main.c
lurongshuang@ubuntu:~/work/work1020$ 

fpic的目的是什么?(赵晨斌讲师的理解)
共享库可能会被不同的进程加载到不同的位置上,如果共享库中的指令使用了绝对地址、外部模块地址,那么在共享库被加载时就必须根据相关模块的加载位置对这个地址做调整,也就是修改这些地址,让它在对应进程中能正确访问,而被修改到的段就不能实现多进程共享一份物理内存,它们在每个进程中都必须有一份物理内存的拷贝。fPIC指令就是为了让使用到同一个共享对象的多个进程能尽可能多的共享物理内存,它背后把那些涉及到绝对地址、外部模块地址访问的地方都抽离出来,保证代码段的内容可以多进程相同,实现共享。

4.生成动态库

gcc/cc -shared 目标文件(xxx.o) -o lib库名.so
gcc -shared add.o -o libadd.so
lurongshuang@ubuntu:~/work/work1020$ gcc -shared add.o -o libadd.so
lurongshuang@ubuntu:~/work/work1020$ ls
add.c  add.h  add.o  libadd.so  main.c
lurongshuang@ubuntu:~/work/work1020$ 

5.编写测试文件(main.c)

#include <stdio.h>
#include "add.h"

int main(){
    printf("1+1=%d",add(1,1));
    return 0;
}

6.使用动态库文件

动态链接两种不同的方式:
1:隐式链接
隐式链接在编译/链接阶段完成,由编译系统根据动态库的头文件和库文件进行编译和链接,从而确定待调用的函数原形和地址。
2.显式链接
显式链接则是利用API函数实现加载和卸载共享库,获取带调用函数地址,获取错误信息等功能。(动态加载共享库)

第一种使用方式:直接链接 (隐式链接)

gcc 目标文件  动态库文件
gcc main.o libadd.so  
lurongshuang@ubuntu:~/work/work1020$ ls
add.c  add.h  add.o  libadd.so  main.c  main.o
lurongshuang@ubuntu:~/work/work1020$ gcc main.o libadd.so
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
./a.out: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory

----------------------------------注意下面这段 --------------------------------

直接运行是不能成功的 需要配置 LD_LIBRARY_PATH环境变量

lurongshuang@ubuntu:~/work/work1020$ export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:.
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$ 

----------------------------------注意上面这段 --------------------------------

第二种使用方式:使用编译选项进行链接(掌握) (隐式链接)

gcc/cc main.o -l 库名 -L 库文件所在的路径 
gcc main.o -l add -L .
lurongshuang@ubuntu:~/work/work1020$ ls
add.c  add.h  add.o  a.out  libadd.so  main.c  main.o
lurongshuang@ubuntu:~/work/work1020$ rm a.out 
lurongshuang@ubuntu:~/work/work1020$ gcc main.o -l add -L .
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$ 

第三种使用方式:配置环境变量LIBRARY_PATH(隐式链接)

export LIBRARY_PATH=路径
gcc 目标文件 -l 库名
export LIBRARY_PATH=$LIBRARY_PATH:.
gcc main.o -l add
lurongshuang@ubuntu:~/work/work1020$ ls
add.c  add.h  add.o  libadd.so  main.c  main.o
lurongshuang@ubuntu:~/work/work1020$ export LIBRARY_PATH=$LIBRARY_PATH:.
lurongshuang@ubuntu:~/work/work1020$ gcc main.o -l add
lurongshuang@ubuntu:~/work/work1020$ ./a.out 
1+1=2
lurongshuang@ubuntu:~/work/work1020$ 

第四种使用方式:使用函数动态加载(显式加载)

void *dlopen(const  char  *filename,int flag);

需要引入头文件 dlfcn.h
第一个参数:字符串形式的共享库文件名
第二个参数:标志 RTLD_LAZY -懒加载 只适用于函数。只有在函数被执行的时候,才确定函数的地址。函数不执行,不加载。RTLD_NOW -立即加载 在dlopen返回之前,动态库的符号就已经确定了地址
返回值:通用类型指针, 成功返回句柄,暂时理解为首地址,失败返回NULL
函数功能:主要用于打开和加载共享库文件

char *dlerror(void);

函数功能:主要用于获取dlopen等函数调用过程发生的最近一个错误的详细信息,返回NULL则表示没有错误发生

void *dlsym(void *handle,const char *symbol);

第一个参数:句柄,也就是dlopen函数的返回值
第二个参数:字符串形式的符号,表示函数名
返回值:成功返回函数在内存中的地址,失败返回NULL
函数功能: 主要用于根据句柄和函数名获取在内存中的地址

int dlclose(void *handle);

函数功能: 主要用于关闭参数handle所指定的共享库,成功返回0,失败返回非0,当共享库不再被任何程序使用时,则回收共享库所占用的内存空间

结合以上罗列函数,综合演示如下:

#include <stdio.h>
#include <dlfcn.h> 
//简单写一下  忽略部分语法错误
int main(){
    int (*add)(int,int);
    void *handle = dlopen("./libadd.so",RTLD_NOW);
    if(handle==NULL) {
        char * str = dlerror();
        if(str!=NULL) {
            printf("出现问题:%s",str);          
        }
        return -1;
    }
    add = dlsym(handle,"add");
    if(add==NULL) {
        char * str = dlerror();
        if(str!=NULL) {
            printf("出现问题:%s",str);          
        }
        dlclose(handle);
        return -1;
    }
    printf("1+1=%d\n",add(1,1));
    dlclose(handle);
    return 0;
}

补充

列举一下生成库文件过程中 2个辅助的命令
1.查看目标文件所依赖的动态库

ldd  XXX

例如:lld a.out
2.查看目标文件所包含的静态库文件、动态库文件、函数名称、变量等。

nm XXX

例如:nm libadd.o

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容

  • 前言 1.静态库和动态库有什么异同? 静态库:链接时完整地拷贝至可执行文件中,被多次使用就有多份冗余拷贝。利用静态...
    Ly梦k阅读 8,580评论 3 18
  • 一、温故而知新 1. 内存不够怎么办 内存简单分配策略的问题地址空间不隔离内存使用效率低程序运行的地址不确定 关于...
    SeanCST阅读 7,808评论 0 27
  • 1.什么是库在windows平台和linux平台下都大量存在着库。本质上来说库是一种可执行代码的二进制形式,可以被...
    鹰击司马阅读 1,120评论 0 2
  • 一、简介 实际开发工程中,一般会有很多函数只是声明,而找不到实现的代码,因为那些实现代码已经编译成库了。在Linu...
    konishi5202阅读 3,813评论 0 5
  • ORA-00001: 违反唯一约束条件 (.) 错误说明:当在唯一索引所对应的列上键入重复值时,会触发此异常。 O...
    我想起个好名字阅读 5,317评论 0 9