Linux-(C/C++)动态链接库生成以及使用(libxxx.so)

Linux中so文件为共享库,与windows下dll类似,不过实现要简单。

so可以供多个进程使用,不同进程调用同一个so文件,所使用so文件不同。

so文件源程序不需要main函数,有也不会被执行。

下面通过一个简单例子,来学习.so文件的制作跟使用(前提已经配置好环境)、

主要了解学习用C语言编译一个动态链接库,如何使用这个库

1、通过简单max函数,生成一个libmax.so链接库

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * max.c 
  3.  * 
  4.  *  Created on: 2016年7月24日 
  5.  *      Author: Andy_Cong 
  6.  */  
  7.   
  8. /*生成libmax.so链接库*/  
  9. /* 
  10.  * # -shared 为链接库  让编译器知道是要编译一个共享库 
  11.  * # -fPIC(Position Independent Code)   编译生成代码与位置无关 
  12.  */  
  13. int max(int a,int b)  
  14. {  
  15.     return a>b?a:b;  
  16. }  
  17. /* 
  18.  * gcc -Wall -g  -fPIC -c max.c -o max.o 
  19.  * gcc -shared max.o -o libmax.so 
  20.  * -g -Wall 供调试使用,不是必须的 
  21.  */  

编译结果:生成libmax.so


2、使用链接库,需要包含头文件(很正常,我们平时使用C库函数也需要包含相关头文件)

max.h头文件如下

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * max.h 
  3.  * 
  4.  *  Created on: 2016年7月24日 
  5.  *      Author: Andy_Cong 
  6.  */  
  7.   
  8. #ifndef MAX_H_  
  9. #define MAX_H_  
  10.   
  11. int max(int a,int b);  
  12.   
  13.   
  14. #endif /* MAX_H_ */  

测试函数main.c如下

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * main.c 
  3.  * 
  4.  *  Created on: 2016年7月24日 
  5.  *      Author: Andy_Cong 
  6.  */  
  7. #include<stdio.h>  
  8. #include"max.h"  
  9. int main(void)  
  10. {  
  11.     printf("call max function results is: %d\n",max(1,1));  
  12.     return 0;  
  13. }  
  14. /* 使用libmax.so库 
  15.  * gcc -o  main main.c -L. -lmax 
  16.  * 
  17.  *-L.: 在当前路径下寻找.so文件 
  18.  *-lmax: 要链接这个libmax.so文件 
  19.  * 
  20.  * */  

运行结果:生成可执行程序main(成功了)



3、使用C++编译使用C语言提供的链接库, 编译链接出错(下面只是简单将main.c 改为main.cpp)



怎么办呢??


libmax这个库仅适合C使用,C++并不适合,如果想编译一个可以供C++使用。那么头文件(max.h)就需要改变,

需要额外增加一句:extern "C"

max.h(修改如下)

微笑

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * max.h 
  3.  * 
  4.  *  Created on: 2016年7月24日 
  5.  *      Author: Andy_Cong 
  6.  */  
  7.   
  8. #ifndef MAX_H_  
  9. #define MAX_H_  
  10.   
  11. extern "C"   // 这句话可理解为,告诉编译器,这个动态库(.so)是用C语言写的,  
  12.              // 需要用C语言链接方式来链接这个库,这样就可以g++来编译了。  
  13. int max(int a,int b);  
  14.   
  15. #endif /* MAX_H_ */  

运行结果



这样就解决了。

4、但是这样有一个问题,难道每次编译都要改来改去,有没有同时适合C/C链接库的方法呢?

答案是有的,只需要改动头文件即可,使用条件编译

C++有一个宏:__cpluscplus   当用g++编译的时候,就可以识别这个宏


[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /* 
  2.  * max.h 
  3.  * 
  4.  *  Created on: 2016年7月24日 
  5.  *      Author: Andy_Cong 
  6.  */  
  7.   
  8. /*条件编译*/  
  9. #ifndef MAX_H_  
  10. #define MAX_H_  
  11.   
  12. #ifndef __cpluscplus  
  13. extern "C"  //C++  
  14. {  
  15. #endif  
  16.    int max(int a,int b);  
  17. #ifndef __cplusplus  
  18. }  
  19. #endif  
  20.   
  21. #endif /* MAX_H_ */  

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/509493.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据结构实验图论一:基于邻接矩阵的广度优先搜索遍历

题目描述 给定一个无向连通图&#xff0c;顶点编号从0到n-1&#xff0c;用广度优先搜索(BFS)遍历&#xff0c;输出从某个顶点出发的遍历序列。(同一个结点的同层邻接点&#xff0c;节点编号小的优先遍历&#xff09;输入 输入第一行为整数n&#xff08;0< n <100&#xf…

IO对象流(序列化和反序列化)

序列化和反序列化概念&#xff1a; 序列化: 把堆内存中的java对象数据&#xff0c;通过某种方式把对象存储到磁盘文件中或者传递给其他网络的节点&#xff08;在网络上传输&#xff09; 反序列化: 把磁盘文件中的对象数据或者网络节点上的对象数据&#xff0c;恢复成java对象的…

我是怎么招聘程序员的

http://coolshell.cn/articles/1870.html

数据结构实验之图论二:基于邻接表的广度优先搜索遍历

题目描述 给定一个无向连通图&#xff0c;顶点编号从0到n-1&#xff0c;用广度优先搜索(BFS)遍历&#xff0c;输出从某个顶点出发的遍历序列。(同一个结点的同层邻接点&#xff0c;节点编号小的优先遍历&#xff09;输入 输入第一行为整数n&#xff08;0< n <100&#xf…

IO之打印流

打印流,打印数据的,打印流只能是输出流: PrintStream: 字节打印流 PrintWriter: 字符打印流 -对于PrintWriter来说,当启用字段刷新之后, 调用println或者printf或者format方法,便会立马刷新操作. 如果没有开启自动刷新,则需要手动刷新或者当缓冲区满的时候,再自动刷新. 使…

数据结构实验之查找四:二分查找

题目描述 在一个给定的无重复元素的递增序列里&#xff0c;查找与给定关键字相同的元素&#xff0c;若存在则输出找到的位置,不存在输出-1。 输入 一组输入数据&#xff0c;输入数据第一行首先输入两个正整数n ( n < 10^6 )和m ( m < 10^4 )&#xff0c;n是数组中数据元…

橡皮鸭程序调试法

转自&#xff1a;http://write.blog.csdn.net/postedit 面&#xff0c;让我来为你介绍一个程序调试大法——“橡皮鸭程序调试法”&#xff0c;这个方法在调试界是很出众的&#xff0c;实施起来相当方便和简易&#xff0c;几乎可以随时随地地实验&#xff0c;几乎不需要借助任何…

标准IO概述和操作

标准的IO: 标准的输入: 通过键盘录入数据给程序. 标准的输出: 在屏幕上显示程序数据. 在System类中有两个常量: InputStream in System.in; PrintStream out System.out; 标准流的重定向操作: 标准的输入: 通过键盘录入数据给程序. 重新指定输入的源不再是键盘,而是一个…

十条不错的编程观点

转自&#xff1a;http://coolshell.cn/articles/2424.html 在Stack Overflow上有这样的一个贴子《What’s your most controversial programming opinion?》&#xff0c;翻译成中文就是“你认为最有争议的编程观点是什么&#xff1f;”&#xff0c;不过&#xff0c;在400多个主…

数据结构上机实验之二分查找

题目描述 在一个递增的序列里&#xff0c;查找元素是否存在&#xff0c;若存在输出YES,不存在输出NO.输入 本题多组数据&#xff0c;首先输入一个数字n(n>100000)&#xff0c;然后输入n个数&#xff0c;数据保证数列递增&#xff0c;然后再输入一个查找数字。输出 若存在输出…

IO之 Properties类加载文件

配置文件:资源文件(以.properties作为拓展名的文件)/属性文件: 做项目开发,为何使用配置文件? 把所有的数据存储在代码中,写死了,”硬编码”. 比如:在Java中需要连接数据库,必须拥有数据的账号和密码. 此时我们就得在Java代码中编写,类似的代码: String username”root”…

Makefile系列学习(博客)

http://blog.csdn.net/haoel/article/category/9198/3

数据结构实验之查找六:顺序查找

题目描述 在一个给定的无序序列里&#xff0c;查找与给定关键字相同的元素&#xff0c;若存在则输出找到的元素在序列中的位序和需要进行的比较次数,不存在则输出"No"&#xff0c;序列位序从1到n,要求查找从最后一个元素开始,序列中无重复元素。 输入 连续多组数据输…

IO之数据流

数据流,提供了可以读/写任意数据类型的方法: DataOutputStream: 提供了 writeXxx(xxx value)方法. DataInputStream: 提供了 readXxx()方法. 注意: writeXxx和readXxx必须要对应起来, writeByte写出的数据,此时只能使用readByte读取回来.

可视化的状态机(FSM)

状态机这个概念已经在网上的博客和论坛中都已经说烂了&#xff0c;随便一搜都有一大堆。相关的废话就不多说了&#xff0c;在这里主要是分享一下如何可视化的设计状态机&#xff0c;如何增强项目的灵活性。这里通过一个生活中的电梯来了解一下状态机。 电梯逻辑如下图&#xf…

数据结构上机实验之顺序查找

题目描述 在一个的序列里&#xff0c;查找元素是否存在&#xff0c;若存在输出YES,不存在输出NO.输入 本题多组数据&#xff0c;首先输入一个数字n&#xff0c;然后输入n(n<1000)个数&#xff0c;然后再输入一个查找数字。输出 若存在输出YES,不存在输出NO.示例输入 4 1 3 5…

dll加载问题的解决方法

在使用LoadLibrary() 和LoadLibraryEx()加载dll文件的时候&#xff0c;会产生下面的错误&#xff1a; this application has failed to start because XXXXX.dll was not found. Re-installing the application may fix this problem. 其中xxxxx是你所调用的dll的文件名。 这类…

IO之 随机访问文件(RandomAccessFile)

随机访问文件(RandomAccessFile): 表示可以在该文件的任何位置写出和读取数据. API中文解释&#xff1a; 此类的实例支持对随机访问文件的读取和写入。随机访问文件的行为类似存储在文件系统中的一个大型 byte 数组。存在指向该隐含数组的光标或索引&#xff0c;称为文件指针&…

数据结构实验之排序二:交换排序

题目描述 冒泡排序和快速排序都是基于"交换"进行的排序方法&#xff0c;你的任务是对题目给定的N个&#xff08;长整型范围内的&#xff09;整数从小到大排序&#xff0c;输出用冒泡和快排对这N个数排序分别需要进行的数据交换次数。 输入 连续多组输入数据&#xff…

关于通过dll导出类模板和函数模板

动态链接库中导出模板函数 C 支持函数模板&#xff0c;利用函数模板&#xff0c;可以简化我们的程序代码。我在自己的代码中也经常用到函数模板&#xff0c;但是以前一直以为函数模板是要放到头文件中的&#xff0c;否则调用模板函数时&#xff0c;编译器会找不到函数模板的定义…