[root@kuenking 桌面]# gcc -c -fpic d2.c //编译动态库的方法
[root@kuenking 桌面]# gcc -shared -olibdemo4.so d1.o d2.o //连接动态库
[root@kuenking 桌面]# gcc main.c -ldemo4 -L. -omain //使用动态库的方法,需要demo4库名:-l库名 -L库所在地(.意思是当前目录)
[root@kuenking 桌面]# ldd main
linux-gate.so.1 => (0x00303000)
libdemo4.so (0x006e1000)
libc.so.6 => /lib/libc.so.6 (0x00727000)
/lib/ld-linux.so.2 (0x00701000)
[root@kuenking 桌面]# ./main
10
[root@kuenking 桌面]#
¥¥¥¥¥:d1.c 和d2.c 是两个文件且没有main函数。
问题:
1:执行程序怎么加载动态库?
首先,必须能够找到动态库,然后加载动态库到内存,然后再映射到用户的内存空间(后面的是系统来完成,我们只需要确保可以让其找到动态库,)。
系统对动态库的查找规则:
a: /lib
b: /usr/lib
c:到环境变量LD_LIBRARY_PATH指定的路径。eg: export LD_LIBRARY_PATH=. (.代表当前目录)
2: 动态库没有作为执行程序的一部分,为连接需要他?
连接器需要确定函数在动态库中的位置;
&&&&例子:
d1.c
#include <stdio.h>
int add(int a,int b)
{
return a +b;
}
d2.c
#include <stdio.h>
int mul(int a,int b)
{
return a * b;
}
main.c
#include <stdio.h>
int main()
{
int a = 10;
int b = 2;
scanf("%d%d",&a,&b);
int r = add(a,b);
int k = mul(a,b);
printf("%d,%d\n",r,k);
return 0;
}
如果之前动态库编译连接过,且后期没有修改过,则可以直接按下方编译就可以!
[root@kuenking 桌面]# gcc main.c -ldemo4 -L. -omain
[root@kuenking 桌面]# ./main
这样的就可以了!
[root@kuenking 5351]# cat maps
00110000-0029c000 r-xp 00000000 fd:00 1326114 /lib/libc-2.12.so
0029c000-0029e000 r--p 0018c000 fd:00 1326114 /lib/libc-2.12.so
0029e000-0029f000 rw-p 0018e000 fd:00 1326114 /lib/libc-2.12.so
0029f000-002a2000 rw-p 00000000 00:00 0
00658000-00659000 r-xp 00000000 00:00 0 [vdso]
00701000-0071f000 r-xp 00000000 fd:00 1326103 /lib/ld-2.12.so
0071f000-00720000 r--p 0001d000 fd:00 1326103 /lib/ld-2.12.so
00720000-00721000 rw-p 0001e000 fd:00 1326103 /lib/ld-2.12.so
00769000-0076a000 r-xp 00000000 fd:00 261916 /root/桌面/libdemo4.so
0076a000-0076b000 rw-p 00000000 fd:00 261916 /root/桌面/libdemo4.so
08048000-08049000 r-xp 00000000 fd:00 261767 /root/桌面/main
08049000-0804a000 rw-p 00000000 fd:00 261767 /root/桌面/main
b76fa000-b76fb000 rw-p 00000000 00:00 0
b770f000-b7712000 rw-p 00000000 00:00 0
bfc98000-bfcad000 rw-p 00000000 00:00 0 [stack]
[root@kuenking 5351]#
///
动态库中函数的查找已经封装 成库:libdl.so
dlopen 打开一个动态库
dlsym 在打开动态库查找一个函数
dlclose 关闭动态库
dlerror 返回错误,
###################################】 man dlopen 查看呢
#include <dlfcn.h>
void *dlopen(const char *filename, int flag);
char *dlerror(void);
void *dlsym(void *handle, const char *symbol);
int dlclose(void *handle);
################################
例子:search.c
#include <dlfcn.h>
int main()
{
void *handle = dlopen("./libdemo4.so", RTLD_LAZY);
int (*fun)(int,int) = dlsym(handle,"add");
int sum = fun(8,9);
printf("%d\n",sum);
dlclose(handle);
return 0;
}
编译:
king 桌面]# ldd main
linux-gate.so.1 => (0x00df0000)
libdl.so.2 => /lib/libdl.so.2 (0x008e7000)
libc.so.6 => /lib/libc.so.6 (0x00727000)
/lib/ld-linux.so.2 (0x00701000)
[root@kuenking 桌面]# ./main
17
[root@kuenking 桌面]#
#######################################
make 的命令行使用达到动态库连接编译;
方法:make -f make脚本文件xx.mk 目标文件
1:建立一个:vi demo.mk
demo:
gcc d1.c -c
gcc d2.c -c
gcc d1.o d2.o -shared -olibdemo4.so
gcc main.c -ldemo4 -omain -L.
//注意gcc 前面一定要tab键,不可以有空格键;
[root@kuenking 桌面]# make -f demo.mk
gcc d1.c -c
gcc d2.c -c
gcc d1.o d2.o -shared -olibdemo4.so
gcc main.c -ldemo4 -omain -L.
[root@kuenking 桌面]# ldd main
linux-gate.so.1 => (0x006b6000)
libdemo4.so (0x00961000)
libc.so.6 => /lib/libc.so.6 (0x00727000)
/lib/ld-linux.so.2 (0x00701000)
[root@kuenking 桌面]# ./main
12 13
25,156
jieshu!
6605
##如果ldd main 出现了问题,则一定是demo.mk脚本没有写好!