库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行。库分静态库和动态库两种。

一、静态库和动态库的区别

1. 静态函数库

这类库的名字一般是libxxx.a;利用静态函数库编译成的文件比较大,因为整个 函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了。当 然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译。

2. 动态函数库

这类库的名字一般是libxxx.so;相对于静态函数库,动态函数库在编译的时候 并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数库里的相应函数,因此动态函数库所产生的可执行文件比较小。由于函数库没有被整合进你的 程序,而是程序运行时动态的申请并调用,所以程序的运行环境中必须提供相应的库。动态函数库的改变并不影响你的程序,所以动态函数库的升级比较方便。
linux系统有几个重要的目录存放相应的函数库,如/lib /usr/lib。

二、静态库的使用

静态库的操作工具:gcc和ar 命令。

编写及使用静态库

(1)设计库源码 pr1.c 和 pr2.c

[root@billstone make_lib]# cat pr1.c

void print1()

{

printf("This is the first lib src!\n");

}

[root@billstone make_lib]# cat pr2.c

void print2()

{

printf("This is the second src lib!\n");

}

(2)  编译.c 文件

[bill@billstone make_lib]$ cc -O -c pr1.c pr2.c

[bill@billstone make_lib]$ ls -l pr*.o

-rw-rw-r--        1 bill          bill                    804    4 月  15 11:11 pr1.o

-rw-rw-r--        1 bill          bill                    804    4 月  15 11:11 pr2.o

(3)  链接静态库

为了在编译程序中正确找到库文件,静态库必须按照 lib[name].a 的规则命名,如下例中[name]=pr.

[bill@billstone make_lib]$ ar -rsv libpr.a pr1.o pr2.o

a - pr1.o

a - pr2.o

[bill@billstone make_lib]$ ls -l *.a

-rw-rw-r--        1 bill          bill                  1822    4 月  15 11:12 libpr.a

[bill@billstone make_lib]$ ar -t libpr.a

pr1.o

pr2.o

(4)  调用库函数代码 main.c

[bill@billstone make_lib]$ cat main.c

int main()

{

print1();

print2();

return 0;

}

(5)  编译链接选项

-L 及-l 参数放在后面.其中,-L 加载库文件路径,-l 指明库文件名字.

[bill@billstone make_lib]$ gcc -o main main.c -L./ -lpr

[bill@billstone make_lib]$ ls -l main*

-rwxrwxr-x        1 bill          bill                11805    4 月  15 11:17 main

-rw-rw-r--         1 bill          bill                      50    4 月  15 11:15 main.c

(6)执行目标程序

[bill@billstone make_lib]$ ./main

This is the first lib src!

This is the second src lib!

[bill@billstone make_lib]$

三、动态库的使用

编写动态库

(1)设计库代码

[bill@billstone make_lib]$ cat pr1.c

int p = 2;

void print()

{

printf("This is the first dll src!\n");

}

[bill@billstone make_lib]$

(2)生成动态库

[bill@billstone make_lib]$ gcc -O -fpic -shared -o dl.so pr1.c

[bill@billstone make_lib]$ ls -l *.so

-rwxrwxr-x        1 bill          bill                  6592    4 月  15 15:19 dl.so

[bill@billstone make_lib]$

动态库的隐式调用

在编译调用库函数代码时指明动态库的位置及名字,  看下面实例

[bill@billstone make_lib]$ cat main.c

int main()

{

print();

return 0;

}

[bill@billstone make_lib]$ gcc -o tdl main.c ./dl.so

[bill@billstone make_lib]$ ./tdl

This is the first dll src!

[bill@billstone make_lib]$

当动态库的位置活名字发生改变时,  程序将无法正常运行;  而动态库取代静态库的好处之一则是通过更新动态库而随时升级库的内容.

动态库的显式调用

显式调用动态库需要四个函数的支持,  函数 dlopen 打开动态库,  函数 dlsym 获取动态库中对象基址,  函数 dlerror 获取显式动态库操作中的错误信息,  函数 doclose 关闭动态库.

[bill@billstone make_lib]$ cat main.c

#include <dlfcn.h>

int main()

{

void *pHandle;

void (*pFunc)();                                                    //  指向函数的指针

int *p;

pHandle = dlopen("./d1.so", RTLD_NOW);                  //  打开动态库

if(!pHandle){

printf("Can't find d1.so \n");

exit(1);

}

pFunc = (void (*)())dlsym(pHandle, "print");                //  获取库函数 print 的地址

if(pFunc)

pFunc();

else

printf("Can't find function print\n");

p = (int *)dlsym(pHandle, "p");                                      //  获取库变量 p 的地址

if(p)

printf("p = %d\n", *p);

else

printf("Can't find int p\n");

dlclose(pHandle);                                                                //  关闭动态库

return 0;

}

[bill@billstone make_lib]$ gcc -o tds main.c –ld1 –L.


时还不能立即./tds,因为在动态函数库使用时,会查找/usr/lib、/lib目录下的动态函数库,而此时我们生成的库不在里边。
这个时候有好几种方法可以让他成功运行: 最直接最简单的方法就是把libstr_out.so拉到/usr/lib或/lib中去。 还有一种方法
export LD_LIBRARY_PATH=$(pwd)
另外还可以在/etc/ld.so.conf文件里加入我们生成的库的目录,然后/sbin/ldconfig。
/etc/ld.so.conf是非常重要的一个目录,里面存放的是链接器和加载器搜索共享库时要检查的目录,默认是从/usr/lib
/lib中读取的,所以想要顺利运行,我们也可以把我们库的目录加入到这个文件中并执行/sbin/ldconfig
。另外还有个文件需要了解/etc/ld.so.cache,里面保存了常用的动态函数库,且会先把他们加载到内存中,因为内存的访问速度远远大于硬盘的
访问速度,这样可以提高软件加载动态函数库的速度了。

库依赖的查看

使用ldd命令来查看执行文件依赖于哪些库。

该命令用于判断某个可执行的 binary 档案含有什么动态函式库。
[root@test root]# ldd [-vdr] [filename]
参数说明:
--version                打印ldd的版本号
-v --verbose           打印所有信息,例如包括符号的版本信息
-d --data-relocs      执行符号重部署,并报告缺少的目标对象(只对ELF格式适用)
-r --function-relocs  对目标对象和函数执行重新部署,并报告缺少的目标对象和函数(只对ELF格式适用)
--help                        用法信息。

如果命令行中给定的库名字包含'/',这个程序的libc5版本将使用它作为库名字;否则它将在标准位置搜索库。运行一个当前目录下的共享库,加前缀"./"。

(笔记)Linux下的静态库和动态库使用详解的更多相关文章

  1. (转)Linux下select, poll和epoll IO模型的详解

    Linux下select, poll和epoll IO模型的详解 原文:http://blog.csdn.net/tianmohust/article/details/6677985 一).Epoll ...

  2. 【VB技巧】VB静态调用与动态调用dll详解

    本文“[VB技巧]VB静态调用与动态调用dll详解”,来自:Nuclear'Atk 网络安全研究中心,本文地址:http://lcx.cc/?i=489,转载请注明作者及出处! [[请注意]]:在以下 ...

  3. (总结)Linux下的暴力密码在线破解工具Hydra详解

    (总结)Linux下的暴力密码在线破解工具Hydra详解 学习了:https://blog.csdn.net/yafeichang/article/details/53502869

  4. Linux 下的静态(函数)库、动态(函数)库

    0. 基本 在命名上,静态库的名字一般是 libxxx.a,动态库的名字一般是 libxxx.so,有时 libxxx.so.major.minor,xxx 是该 lib 的名字,major 是主版本 ...

  5. linux下添加动态链接库路径、动态库加载等方法

    linux下添加动态链接库路径的方法 2017年01月20日 10:08:17 阅读数:5596   Linux共享库路径配置 Linux下找不到共享库文件的典型现象为明明已经安装某个软包(如libn ...

  6. linux下项目开发加载动态库:ldconfig与 /etc/ld.so.conf

    场景:自己开发一个项目,程序里包含一些自定义动态库.运行,需要加载这些动态库. 假如这些库在/pro/output/lib/下面,可执行程序在/pro/output/bin/下面. 那么,我们需要: ...

  7. Linux下select, poll和epoll IO模型的详解

    http://blog.csdn.net/tianmohust/article/details/6677985 一).Epoll 介绍 Epoll 可是当前在 Linux 下开发大规模并发网络程序的热 ...

  8. Linux 下Redis集群安装部署及使用详解(在线和离线两种安装+相关错误解决方案)

    一.应用场景介绍 本文主要是介绍Redis集群在Linux环境下的安装讲解,其中主要包括在联网的Linux环境和脱机的Linux环境下是如何安装的.因为大多数时候,公司的生产环境是在内网环境下,无外网 ...

  9. Linux下的crontab定时、执行任务命令详解 oracle 自动备份

    在LINUX中,周期执行的任务一般由cron这个守护进程来处理[ps -ef|grep cron].cron读取一个或多个配置文件,这些配置文件中包含了命令行及其调用时间.cron的配置文件称为&qu ...

随机推荐

  1. lua的模块加载require

    加载指定的模块.首先函数会在 package.loaded 这个表中查看是否已经加载 了 modname 这个模块.如果是,那么 require 会返回保存在 package.loaded[modna ...

  2. linux c编程操作数据库(sqlite3应用)

     首先pThread 不是linux系统默认库,连接的时候需要使用库libpthread.a. 加入-lpthread参数.另外会有lopen什么找不到的情况.加入-ldl 指定目录.Project_ ...

  3. 【转】(五)unity4.6Ugui中文教程文档-------概要-UGUI Interaction Components

    原创至上,移步请戳:(五)unity4.6Ugui中文教程文档-------概要-UGUI Interaction Components 4.Interaction Components 本节涵盖了处 ...

  4. HBase的Write Ahead Log (WAL) —— 整体架构、线程模型【转】

    转自:http://www.cnblogs.com/ohuang/p/5807543.html 解决的问题 HBase的Write Ahead Log (WAL)提供了一种高并发.持久化的日志保存与回 ...

  5. STL之std::set、std::map的lower_bound和upper_bound函数使用说明

    由于在使用std::map时感觉lower_bound和upper_bound函数了解不多,这里整理并记录下相关用法及功能. STL的map.multimap.set.multiset都有三个比较特殊 ...

  6. FIDDLER的使用方法及技巧总结(连载三)FIDDLER使用技巧及方法

    (接上篇!~~~~) 三.FIDDLER使用技巧及方法 1.AutoResponder选项卡的使用 Fiddler的AutoResponder 选项卡允许你使用本地硬盘的文件来作为返回内容,而不是把请 ...

  7. Python爬取猫眼top100排行榜数据【含多线程】

    # -*- coding: utf-8 -*- import requests from multiprocessing import Pool from requests.exceptions im ...

  8. maven pom scope 含义

    maven pom scope 依赖范围控制哪些依赖在哪些classpath 中可用,哪些依赖包含在一个应用中.让我们详细看一下每一种范围: compile (编译范围) compile是默认的范围: ...

  9. SQL Server 数据库自建表

    sysobjects是系统自建的表,里面存储了在数据库内创建的每个对象(约束.默认值.日志.规则.存储过程等). 列名 数据类型 描述 name sysname 对象名 id int 对象标识号 xt ...

  10. 使用jQuery延迟加载js文件

    //异步加载js文件并调用函数 function delayCall(calledFunction, funcParams, jsUrl) { if (eval('typeof '+calledFun ...