gcc生成并且运行动态库
源代码
//main.c
#include"head.h"
#include<stdio.h>
int main()
{
int a=20,b=12;
#ifdef DEBUG
printf("我是一个程序猿,我还会爬树...\n");
#endif
printf("a+b=%d\n",add(a,b));
printf("a-b=%d\n",sub(a,b));
printf("a*b=%d\n",multi(a,b));
printf("a/b=%f\n",div(a,b));
return 0;
}
//head.h
#ifndef _HEAD_H
#define _HEAD_H
#include<stdio.h>
int add(int a,int b);
int sub(int a,int b);
int multi(int a,int b);
double div(int a,int b);
#endif
//operate.c
#include<stdio.h>
int sub(int a,int b)
{
return a-b;
}
int multi(int a,int b)
{
return a*b;
}
double div(int a,int b)
{
return (double)a/b;
}
//add.c
#include<stdio.h>
int add(int a,int b)
{
return a+b;
}
(1)将需要打包成动态库的源文件编译成共享的.o文件
gcc -c -fpic operate.c add.c
这个指令将会生成operate.o和add.o的目标文件,编译的时候如果头文件不在该目录则需要使用-I选项指定头文件的搜索目录。fpic是编译动态库所必须的。
(2)将目标文件打包为共享库
gcc -shared operate.o add.o -o libcalc.so
打包得到的动态库文件名实际是掐头去尾后的calc。
(3)对于打包后的动态库文件发布的话需要同时发布头文件和.so 文件
(4)在使用上述的动态库和头文件的时候,比如将main.c与动态库和头文件放在一个文件夹进行编译时,使用
gcc main.c -L./ -lcalc
这条指令能够正常编译产生a.out文件
(5)运行编译产生的a.out文件。但运行是会报错:
./a.out: error while loading shared libraries: libcalc.so: cannot open shared object file: No such file or directory
解决这个报错有几个途径,可以修改环境变量,或者把动态库放入/usr/lib目录中,以及修改一个配置文件。
对于普通用户而言,只有一个路径可以选择,那就是修改自己的环境变量。
定义环境变量:
export LD_LIBRARY_PATH=/root/gcc06/test1/:$LD_LIBRARY_PATH
这样就相当于增加了动态库的搜索路径。
上面的这条语句可以写到~/.bashrc文件中,则表示永久生效,这适用于安装的软件。也可以在命令行中执行,定义临时的环境变量,可以通过unset删除该环境变量。
(6)定义了环境变量后,代码便能执行成功。
注意
参考静态链接库中的使用方法,在使用动态库编译main.c文件时有另一种方法来编译,不指定环境变量也不能够运行。在编译过程中静态库可以通过这种方式不指定环境变量,动态库就不行,因为静态库在编译时就把库加到程序里面去了,运行时不需要搜索,动态库在运行时必须要搜索。
gcc -Wall main.c libcalc.so
即不通过-l(小写字母l)选项来指定链接库,而是直接使用该目录下的libcalc.so,这样编译之后产生的a.out不可以直接运行。考虑是因为-l选项默认就会到环境变量指定的库或者默认库中去寻找,在运行动态库时无法指定其搜索的路径。
相关的一些指令
ldd a.out
可以查看这个可执行文件能否加载到所需的动态库。
参考学习网站
https://subingwen.cn/linux/