转自:http://www.cnblogs.com/zlcxbb/p/6806269.html

1.什么是静态库

静态库类似windows中的静态lib

关于windows中的静态lib,可参考 Windows动态链接库DLL

特点:包含函数代码声明和实现,链接后所有代码都嵌入到宿主程序中。
只在编译时使用,执行时不再需要该静态库。

2.静态库编写
示例如下:
addvec.c

void addvec(int* x, int* y, int*z, int n)
{
int i=0;
for(; i< n;++i)
z[i] = x[i] + y[i];
}

multvec.c

void multvec(int*x, int* y, int*  z, int n)
{
int i = 0;
for(; i < n; ++i)
z[i] = x[i] * y[i];
}

使用AR工具创建静态库文件:

3.静态库使用
示例如下:
test2.c

#include <stdio.h>
int x[2] = {1, 2};
int y[2] = {3, 4};
int z[2]={0};
int main()
{
addvec(x, y, z, 2);
printf("z = [%d %d]\n", z[0], z[1]);
return 0;
}

编译-链接-运行程序

1)-static参数,表明是静态链接,编译出的是完整的可执行目标文件。
2)当链接器进行链接时,会判断main函数里调用了addvec.o中的addvec函数,
没有调用multvec.o中的任何函数,所以,链接器只会拷贝addvec.o到可执行文件。

4.什么是共享库
共享库类似windows中的动态链接库dll

特点:包含函数代码声明和实现。
只在运行时使用,由动态链接器链接和加载。

根据链接和加载共享库的时机可分为以下两类:
1)自身加载型共享库。
2)运行时加载型共享库

5.自身加载型共享库。
类似windows中的隐式链接
链接时,将共享库的声明信息链接到可执行文件,
应用程序加载时,动态链接库解析声明信息,加载共享库的实现到存储器,重定位应用程序中声明信息到实际地址。

6.自身加载型共享库使用示例
使用-shared参数,指示编译器创建一个共享库。
如下所示,我们创建了一个共享库,并通过自身加载型来使用该共享库。

1)-fPIC参数,指示编译器生成代码无关的代码
2)在链接时,没有拷贝共享库libvec.so的实现,只拷贝了一些重定位和符号表信息
3)程序加载时,动态链接器会解析共享库libvec.so中代码和数据的引用,重定位完成链接任务。
重定位libvec.so的文本和数据到存储器段
重定位p2中引用的libvec.so到以上存储器段
最后链接器将控制传递给程序,此时,共享库的位置就固定了。

7.运行时加载型共享库
类似windows中的显式链接
无需编译时链接,可在运行过程中加载和卸载共享库。

8.运行时加载型共享库使用示例
Linux提供了一组运行过程中加载和卸载共享库的API,如下所示:
#include<dlfcn.h>

/* 加载和链接共享库 filename
filename:共享库的名字
flag有:RTLD_LAZY, RTLD_NOW,二者均可以和RTLD_GLOBAL表示取或
*/
void *dlopen(const char *filename, int flag); // 若成功则返回执行句柄的指针,否则返回NULL /*根据共享库操作句柄与符号,返回符号对应的地址
handle:共享库操作句柄
symbol:需要引用的符号名字
*/
void *dlsym(void *handle, char *symbol); // 若成功则返回执行符号的指针(即地址),若出错则返回NULL /* 如果没有程序正在使用这个共享库,卸载该共享库 */
int dlclose(void *handle); // 若卸载成功,则返回0,否则返回-1 /* 捕捉最近发生的错误 */
const char *dlerror(void); // 若前面对dlopen,dlsym或dlclose调用失败,则返回错误消息,否则返回NULL

根据以上API,我们可以方便地加载和卸载共享库,如下所示:

#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h> int x[2] = {1, 2};
int y[2] = {3, 4};
int z[2] ={0}; int main()
{
void *handle;
void (*addvec)(int *, int *, int *,int);
char *error; handle = dlopen("./libvector.so", RTLD_LAZY);
if(!handle){
fprintf(stderr, "%s\n", dlerror());
exit(1);
} addvec = dlsym(handle, "addvec");
if((error = dlerror()) != NULL){
fprintf(stderr, "%s\n", dlerror());
exit(1);
} addvec(x, y, z, 2);
printf("z = [%d %d]\n", z[0], z[1]); if(dlclose(handle) < 0){
fprintf(stderr, "%s\n", dlerror());
exit(1);
} return 0;
}

运行程序:

其中,-ldl参数,表示程序运行时需要用到共享库

参考:http://www.cnblogs.com/shijingjing07/p/5608104.html

Linux静态库和共享库【转】的更多相关文章

  1. Linux 静态库与共享库的使用

    申明: 正如题如示,本篇讲的是Linux下是静态库与共享库,而Window下的动态链接库详细情况可见这篇文章:windows动态链接库 DLL 浅析.虽然原理,思想差不多,但是细节却各有不同. 一.静 ...

  2. 【转】Linux 静态库与共享库的使用

    原文网址:http://blog.csdn.net/heyabo/article/details/11688517 申明: 正如题如示,本篇讲的是Linux下是静态库与共享库,而Window下的动态链 ...

  3. linux下的静态库和共享库

    转载&&增加:      我们在编写一个C语言程序的时候,经常会遇到好多重复或常用的部分,如果每次都重新编写固然是可以的,不过那样会大大降低工作效率,并且影响代码的可读性,更不利于后期 ...

  4. Linux下库的制作(静态库与共享库)

    库中实际上就是已编译好的函数代码,可以被程序直接调用. Linux下的库一般的位置在/lib或者/usr/lib中 静态库 静态库是复制拷贝到调用函数中的,函数运行的时候不再需要静态库,因为静态库是在 ...

  5. 第二课 GCC入门之静态库以及共享库

    序言: 前面一课讲了gcc的简单入门,包括gcc编译步骤:预处理:编译:汇编:链接.今天这节课就来讲下linux的库也欢迎大家吐糟共同学习. 原理: linux系统中分为2种库:静态库和共享库.静态库 ...

  6. Linux下Qt创建共享库与链接共享库详解

    随着程序写的逐渐变多,或多或少的我们都会使用别人写好的库:或者我们不想让别人看到我们的一些核心程序,可以将核心程序封装成库.本次和大家分享的是在Ubuntu下使用Qt生成共享库以及在Qt中链接共享库的 ...

  7. Linux静态库和共享库

    1.什么是静态库静态库类似windows中的静态lib 关于windows中的静态lib,可参考 Windows动态链接库DLL 特点:包含函数代码声明和实现,链接后所有代码都嵌入到宿主程序中. 只在 ...

  8. Linux命令(十二)制作静态库和共享库

    1. 静态库 静态库文件命名:libxxxx.a 1.1 步骤: ar rcs libCalc.a *.o 1.2 用nm查看文件内容 1.3 发布并使用 gcc main.c -o mycpp.ou ...

  9. gcc创建静态库和共享库

    静态库和动态(共享)库静态库:编译程序在编译使用库提供的功能代码的程序时将代码复制到该程序然后编译成可执行程序,这种库成为静态库共享库:共享库比静态库的处理方式更加灵活,因而其产生的可执行文件更小,其 ...

随机推荐

  1. TScreen 类 - 通过 Screen 更换光标

    //更换窗体或某个控件的光标可以不通过 Screen 对象, 譬如: begin   Self.Cursor := crAppStart;   Panel1.Cursor := crHandPoint ...

  2. asp.net MVC4在Action间跳转 RedirectToAction 传值参数问题

    return RedirectToAction("Test", new { cw = cw, firstdirectoryid = firstdirectoryid }); 上式中 ...

  3. 【bzoj1798】[Ahoi2009]Seq 维护序列seq 线段树

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  4. 【刷题】洛谷 P3804 【模板】后缀自动机

    题目描述 给定一个只包含小写字母的字符串 \(S\) , 请你求出 \(S\) 的所有出现次数不为 \(1\) 的子串的出现次数乘上该子串长度的最大值. 输入输出格式 输入格式: 一行一个仅包含小写字 ...

  5. [bzoj4398] 福慧双修 最短路 二进制分组

    ---题面--- 题解: 考场上看的这道题,,,当时70分算法打挂了,今天才知道这个也是原题.... 首先,对于不跟1相邻的边,肯定不会经过两次,因为经过两次就回来了,除了增加路径长度之外没有任何意义 ...

  6. CF97B:Superset——题解

    http://codeforces.com/problemset/problem/97/B 题目大意:给n个点,添加一些点,使得任意两个点: 1.在同一条线上 2.以它们为顶点构成的矩形上有其他点. ...

  7. [Leetcode] Sum root to leaf numbers求根到叶节点的数字之和

    Given a binary tree containing digits from0-9only, each root-to-leaf path could represent a number. ...

  8. Python3简单入门

    在Mac和Linux上运行Python时,请打开终端,然后运行python3 Mac OSX 正确地同时安装Python 2.7 和Python3:  http://www.jianshu.com/p ...

  9. 51nod 1275 连续子段的差异(twopointer+单调队列)

    对于每一个i找到最近的j满足最大值-最小值>K,对答案的贡献为j-i,用单调队列维护最值即可 #include<iostream> #include<cstdlib> # ...

  10. LoadRunner中的IP欺骗

    应用程序服务器和网络设备使用IP地址来识别客户端.应用程序服务器通常会对来自同一计算机的客户端信息进行高速缓存. 网络路由器尝试对原信息和目标信息进行高速缓存以优化吞吐量.如果多个用户具有相同的IP地 ...