库的分类

根据链接时期的不同,库分为静态库和动态库之分。

静态库:在链接阶段被链接的,生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行。

动态库:在程序执行的时候被链接的,即使程序编译完,库仍须保留在系统上,以供程序运行时调用。

动态链接库

一、隐式调用

    --> 创建动态链接库

#include<stdio.h>
void hello()
{
printf("hello world/n");
}

编译生成动态库libhello.so:

gcc -fPIC -shared hello.c -o libhello.so

-shared 该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件;

-fpic 选项表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。

    -->  再编辑一个测试文件test.c,内容如下

#include<stdio.h>
int main()
{
printf("call hello()");
hello();
}

编译:

gcc test.c -lhello -L. -o test

-l 选项告诉编译器要使用hello这个库。奇怪的地方是动态库的名字是libhello.so,这里却使用hello;-L 选项告诉编译器在当前目录中查找库文件。

    -->  编译成功后执行./test, 仍然出错,说找不到库

有两种方法:
  
1、可以把当前路径加入 /etc/ld.so.conf中然后运行ldconfig,或者以当前路径为参数运行ldconfig(要有root权限才行)。
  2、把当前路径加入环境变量LD_LIBRARY_PATH中, 如:export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH

二、显式调用

  显式调用需要包含头文件#include <dlfcn.h>。涉及到下面几个函数:dlopen()、dlsym()、dlerror()、dlclose()。

dlopen()  // 函数以指定模式打开指定的动态链接库文件,并返回一个句柄给dlsym()的调用进程。
dlcolse() // 用来卸载打开的库。
dlerror() // 当动态链接库操作函数执行失败时,可以返回出错信息,返回值为NULL时表示操作函数执行成功。

编译时候要加入 -ldl (指定dl库)

具体的函数原型如下:

void *dlopen(const char *filename, int flag);
char *dlerror(void);
void *dlsym(void *handle, const char *symbol);
int dlclose(void *handle);

调用示例:

#include <stdio.h>
#include <stdlib.h>
include <dlfcn.h> //动态链接库路径
#define LIB_CACULATE_PATH "./libcaculate.so" //函数指针
typedef int (*CAC_FUNC)(int, int); int main()
{
void *handle;
char *error;
CAC_FUNC cac_func = NULL; //打开动态链接库
handle = dlopen(LIB_CACULATE_PATH, RTLD_LAZY);
if (!handle) {
fprintf(stderr, "%s\n", dlerror());
exit(EXIT_FAILURE);
} //清除之前存在的错误
dlerror(); //获取一个函数
*(void **) (&cac_func) = dlsym(handle, "add");
if ((error = dlerror()) != NULL) {
fprintf(stderr, "%s\n", error);
exit(EXIT_FAILURE);
}
printf("add: %d\n", (*cac_func)(,)); cac_func = (CAC_FUNC)dlsym(handle, "sub");
printf("sub: %d\n", cac_func(,)); cac_func = (CAC_FUNC)dlsym(handle, "mul");
printf("mul: %d\n", cac_func(,)); cac_func = (CAC_FUNC)dlsym(handle, "div");
printf("div: %d\n", cac_func(,)); //关闭动态链接库
dlclose(handle);
exit(EXIT_SUCCESS);
}

静态链接库

仍使用刚才的hello.c和test.c。
  1. gcc -c hello.c 注意这里没有使用-shared选项
  2. 把目标文件归档 :

 ar -r libhello.a hello.o

    程序 ar 配合参数 -r 创建一个新库 libhello.a 并将命令行中列出的对象文件插入。采用这种方法,如果库不存在的话,参数 -r 将创建一个新的库,而如果库存在的话,将用新的模块替换原来的模块。
  3. 在程序中链接静态库:

  gcc test.c -lhello -L. -static -o hello.static
// 或者
gcc test.c libhello.a -L. -o hello.static

生成的hello.static就不再依赖libhello.a了LD_LIBRARY_PATH

静态库和动态库的比较

静态库被链接后库直接嵌入到可执行文件中,缺点:

  1、造成系统空间被浪费。如果多个程序链接了同一个库,则每一个生成的可执行文件就都会有一个库的副本,必然会浪费系统空间。

  2、一旦发现了库中有bug,挽救起来就比较麻烦。必须一一把链接该库的程序找出来,然后重新编译。

动态库是在程序运行时被链接的,弥补上述不足:

  1、磁盘上只须保留一份副本,因此节约了磁盘空间。

  2、如果发现了bug或要升级也很简单,只要用新的库把原来的替换掉就行了。

linux --> 动态库和静态库的更多相关文章

  1. Linux中的动态库和静态库(.a/.la/.so/.o)

    Linux中的动态库和静态库(.a/.la/.so/.o) Linux中的动态库和静态库(.a/.la/.so/.o) C/C++程序编译的过程 .o文件(目标文件) 创建atoi.o 使用atoi. ...

  2. Linux系统中“动态库”和“静态库”那点事儿 /etc/ld.so.conf 动态库的后缀为*.so 静态库的后缀为 libxxx.a ldconfig 目录名

    Linux系统中“动态库”和“静态库”那点事儿 /etc/ld.so.conf  动态库的后缀为*.so  静态库的后缀为 libxxx.a   ldconfig   目录名 转载自:http://b ...

  3. Linux系统中“动态库”和“静态库”那点事儿【转】

    转自:http://blog.chinaunix.net/uid-23069658-id-3142046.html 今天我们主要来说说Linux系统下基于动态库(.so)和静态(.a)的程序那些猫腻. ...

  4. linux中C的静态库和动态库分析

    从开始学C语言写第一个"hello world"历程到现在,我依然困惑于到底这个程序完整的执行流程是什么样的.不过,现在我正在尝试一点一点的揭开它的面纱.现在,我尝试分析linux ...

  5. 技巧:Linux 动态库与静态库制作及使用详解

    技巧:Linux 动态库与静态库制作及使用详解 标准库的三种连接方式及静态库制作与使用方法 Linux 应用开发通常要考虑三个问题,即:1)在 Linux 应用程序开发过程中遇到过标准库链接在不同 L ...

  6. Linux系统中“动态库”和“静态库”那点事儿

    摘自http://blog.chinaunix.net/uid-23069658-id-3142046.html 今天我们主要来说说Linux系统下基于动态库(.so)和静态(.a)的程序那些猫腻.在 ...

  7. Linux的动态库与静态库

    1.动态库与静态库简介 在实际的软件开发中,为了方便使用一些被重复调用的公共代码,我们经常将这些公共的函数编译成动态库或静态库.我们知道程序一般要经过预处理.编译.汇编和链接这几个步骤才能变成可执行的 ...

  8. linux动态库与静态库混合连接

      1, 在应用程序需要连接外部库的情况下,linux默认对库的连接是使用动态库,在找不到动态库的情况下再选择静态库.使用方式为: gcc test.cpp -L. -ltestlib 如果当前目录有 ...

  9. Linux环境编译动态库和静态库总结

    对Linux环境动态库和静态库的一些基础知识做一些总结, 首先总结静态库的编译步骤. 1 先基于.cpp或者.c文件生成对应的.o文件 2将几个.o文件 使用ar -cr命令 生成libname.a文 ...

  10. linux系统环境下的静态库和动态库的制作

    linux系统下的应用编程需要系统提供的库文件,包括静态库或动态库.不管是静态库还是动态库,都是编译好的二进制文件.在我们编译程序时要链接的目标文件,静态库是链接的时候直接编译到程序里,和程序成为一体 ...

随机推荐

  1. 单片机:STC89C52的最小单元

    STC89C52RC是STC公司生产的一种低功耗.高性能CMOS8位微控制器,具有 8K字节系统可编程Flash存储器.STC89C52使用经典的MCS-51内核,但做了很多的改进使得芯片具有传统51 ...

  2. linux下的framebuffer显示图片

    void  showbmp2()  {      int x,y;      unsigned char *p;      int index=0;      struct fb_var_screen ...

  3. 一款PHP环境整合工具—VertrigoServ介绍

    Vertrigo简介 VertrigoServ 是一个Windows平台下的非常专业的.易于安装的免费网络开发环境,它集成了Apache, PHP, MySQL, SQLite, SQLiteMana ...

  4. 关于vue-axios的post方式,后台无法解析传参问题

    启用jq方法更改数字格式 var params = {'addid':item.addid}; var str = $.param(params);

  5. C语言 字符串前加L的意义 如:L“A”

    转自:http://c.biancheng.net/cpp/html/1069.html Unicode或者宽字符都没有改变char数据型态在C中的含义.char继续表示1个字节的储存空间,sizeo ...

  6. layer与iframe 间值的相互传递

    在A页添加 <input id="handle_status" value="" hidden="hidden"> jq代码 l ...

  7. 【ZJOI2008】树的统计(树链剖分)

    题面 Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II ...

  8. bzoj2157

    LCT板子,打个lazy即可 # include <stdio.h> # include <stdlib.h> # include <iostream> # inc ...

  9. [BZOJ4034] [HAOI2015] T2 (树链剖分)

    Description 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子树中所 ...

  10. [BZOJ3207] 花神的嘲讽计划Ⅰ (主席树)

    Description 背景 花神是神,一大癖好就是嘲讽大J,举例如下: “哎你傻不傻的![hqz:大笨J]” “这道题又被J屎过了!!” “J这程序怎么跑这么快!J要逆袭了!” …… 描述 这一天D ...