Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2566172
  • 博文数量: 315
  • 博客积分: 3901
  • 博客等级: 少校
  • 技术积分: 3640
  • 用 户 组: 普通用户
  • 注册时间: 2011-05-08 15:32
个人简介

知乎:https://www.zhihu.com/people/monkey.d.luffy Android高级开发交流群2: 752871516

文章分类

全部博文(315)

文章存档

2019年(2)

2018年(1)

2016年(7)

2015年(32)

2014年(39)

2013年(109)

2012年(81)

2011年(44)

分类: LINUX

2011-08-09 18:45:13

                            linux下动态库的创建和使用
     具体也不清楚照着资料做一遍有没有用。但是还是做了,主要原因是因为在linux下编译程序的时候会提到动态库的概念。老师也提到了。所以就想跑一遍算了。管他有没有用呢???
     具体步骤就是参照网上的做,注意文件中代码的格式和编译的格式就好了。所以自己稍微修改了简化了下。但也不是我的。呵呵。
    下面开始吧:(linux下动态库的创建和使用)
    ----内容转载,代码自己敲了一遍。能看懂点。唉!(文件名注意点哦??)
 
动态链接库应用示范
     我准备编写两个函数,一个用于查询当前日期getdate,一个用于查询当前时间gettime,并将这两个
    函数存于动态链接库my.so中。为此,需要做以下几项工作。
    1.1 编写用户接口文件datetime.h,内容如下(每行前面的数字为行号):
/*datatime.h: LEGO
*This is my first dynamic library
*/
#ifndef _DATATIME_H_
#define _DATATIME_H_
typedef struct{
 int year;
 int mon;
 int day;
}DATETYPE;
typedef struct{
 char hour;
 char min;
 char sec;
}TIMETYPE;
#ifdef SHARED
int (*getdate)(DATETYPE *d);
#else
int getdate(DATETYPE *d);
#endif
#ifdef SHARED
int (*gettime)(TIMETYPE *t);
#else
int gettime(TIMETYPE *t);
#endif
#endif
这个用户接口文件中,先定义了日期与时间结构,接着定义一下函数的原型。动态函数与静态函数的原型说明不同的是,动态函数应使用(*函数名)的形式,以便引用其指针。若要引用文件中的动态函数说明,用户应该定义一下SHARED宏,这样才能使用。
1.2 编写getdate.c,源程序如下:
 #include
#include "datatime.h"
int getdate(DATETYPE *d)
{
 long ti;
 struct tm *tm;
 time(&ti);
 tm = localtime(&ti);
 d->year = tm->tm_year + 1900;
 d->mon  = tm->tm_mon + 1;
 d->day  = tm->tm_mday;
}
在getdate函数中,先调用time取得以秒计的系统时间,再用localtime函数转换一下时间结构,最后调整得到正确的日期。
 
1.3 编写gettime.c,源程序如下:
#include
#include "datatime.h"
int gettime(TIMETYPE *t)
{
 long ti;
 struct tm *tm;
 time(&ti);
 tm = localtime(&ti);
 t->hour = tm->tm_hour;
 t->min  = tm->tm_min;
 t->sec  = tm->tm_sec;
}
gettime函数与getdate函数相仿,先用time函数取得以秒计的系统时间,再用localtime函数转换一下时间结构,最后返回当前的时间(不需调整)。
 
1.4 编写维护文件makefile-lib,内容如下:
 all:my.so
src = getdate.c gettime.c
TGT = $(src:.c=.o)
$(src):datatime.h
 @touch $@
%.o:%.c
 cc -c $?
my.so:$(TGT)
 cc -shared -o $@ $(TGT)
编写维护文件的目的,在于方便程序员维护程序,尤其是维护比较大的工程项目。一个素质良好的程序员应该学会熟练地编写维护文件makefile。定义了文件间的依赖关系后,一旦源文件发生变化,仅需make一下,其目标文件维护代码会自动执行,从而自动更新目标文件,减少了许多工作量。注意: 每行维护代码必须以TAB(跳格键)开始,不是的话make时将出错。
本维护文件第1行是注释行,以#号开头;文件第3行定义所有需要维护的函数库;第5行定义相关源程序文件;第7行定义目标文件;第9-10行说明所有源程序依赖于datetime.h头文件,并有相应维护代码,即touch一下,更新一下源文件的时间;第12-13行定义.o文件依赖于相应的.c文件,并指定了维护代码,即用cc编译一下;第16-17行定义共享库my.so依赖的目标文件,维护代码中用-shared编译选项,以生成动态链接库my.so。
编写维护文件的目的,在于方便程序员维护程序,尤其是维护比较大的工程项目。一个素质良好的程序员应该学会熟练地编写维护文件makefile。定义了文件间的依赖关系后,一旦源文件发生变化,仅需make一下,其目标文件维护代码会自动执行,从而自动更新目标文件,减少了许多工作量。注意: 每行维护代码必须以TAB(跳格键)开始,不是的话make时将出错。
本维护文件第1行是注释行,以#号开头;文件第3行定义所有需要维护的函数库;第5行定义相关源程序文件;第7行定义目标文件;第9-10行说明所有源程序依赖于datetime.h头文件,并有相应维护代码,即touch一下,更新一下源文件的时间;第12-13行定义.o文件依赖于相应的.c文件,并指定了维护代码,即用cc编译一下;第16-17行定义共享库my.so依赖的目标文件,维护代码中用-shared编译选项,以生成动态链接库my.so。
 
1.5 运行make -f makefile-lib 命令
 make运行后,动态链接库my.so就产生了,我们就可以在程序中调用了。如果想让系统所有用户都可以使用,则应以root用户登录系统,将这个库拷贝到/lib目录下(命令:cp my.so /lib),或者在/lib目录下建个符号连接即可(命令:ln -s `pwd`/my.so /lib)。这里我做了一个软链接(这样不太好,万一你删了,那库就不能用了,最好拷贝),就是将我生成的my.so做一个映射,映射到/lib目录下,你就会看到/lib目录下有个my.so.
 
 程序范例
 下面的程序装载了动态链接库my.so,并用getdate,gettime取得当前日期与时间后输出。
 //dy.c
#include
#include
#include
#define SOFILE "/lib/my.so"  //这里指定我自己的库
#define SHARED
#include "datatime.h"
int main(int argc, char* argv[])
{
 DATETYPE d;
 TIMETYPE t;
 void *dp;
 char *error;
 puts("Dynamic examples!");
 dp = dlopen(SOFILE, RTLD_LAZY);
 if(dp == NULL)
 {
  fputs(dlerror(), stderr);
  exit(1);
 }
 getdate = dlsym(dp, "getdate");
 error = dlerror();
 if(error)
 {
  fputs(error, stderr);
  exit(1);
 }
 getdate(&d);
 printf("Current date: %04d-%02d-%02d\n",d.year, d.mon, d.day);
 gettime = dlsym(dp, "gettime");
 error = dlerror();
 if(error)
 {
  fputs(error, stderr);
  exit(1);
 }
 gettime(&t);
 printf("Current time: %02d:%02d:%02d\n", t.hour, t.min, t.sec);
 dlclose(dp);
 return 0;
}
程序说明:
第8行: 包含标准输入输出头文件,因为程序中使用了printf,puts,fputs等标准输入输出函数,需要让编译器根据头文件中函数的原型,检查一下语法;
第10-11行: 包含动态链接库功能头文件,并定义动态链接库名称;
第13-14行: 定义宏SHARED以便引用14行的头文件datetime.h中的动态函数说明;
第25行: 用dlopen打开SOFILE共享库,返回句柄dp;
第27-31行: 检测dp是否为空,为空则显示错误后退出;
第33行: 用dlsym取得getdate函数动态地址;
第35-40行: 如果dlerror返回值不为空,则dlsym执行出错,程序显示错误后退出;
第42-43行: 执行getdate调用,输出当前日期;
第45行: 用dlsym取得gettime函数动态地址;
第47-52行: 如果dlerror返回值不为空,则dlsym执行出错,程序显示错误后退出;
第54-55行: 执行gettime调用,输出当前时间;
第57行: 用dlclose关闭dp所指示的动态链接库;
第59行: 程序退出,返回0值。
LINUX创建与使用动态链接库并不是一件难事。
编译函数源程序时选用-shared选项即可创建动态链接库,注意应以.so后缀命名,最好放到公用库目录(如/lib,/usr/lib等)下面,并要写好用户接口文件,以便其它用户共享。
使用动态链接库,源程序中要包含dlfcn.h头文件,写程序时注意dlopen等函数的正确调用,编译时要采用-rdynamic选项与-ldl选项,以产生可调用动态链接库的执行代码。
编译函数源程序时选用-shared选项即可创建动态链接库,注意应以.so后缀命名,最好放到公用库目录(如/lib,/usr/lib等)下面,并要写好用户接口文件,以便其它用户共享。
使用动态链接库,源程序中要包含dlfcn.h头文件,写程序时注意dlopen等函数的正确调用,编译时要采用-rdynamic选项与-ldl选项,以产生可调用动态链接库的执行代码。
3.做好以上步骤后。我们就可以执行这个测试文件了。当然可以编写一个makefile来执行。
 我试了下,需要把datatime.h也放在测试文件目录下。然后执行编译命令(注意参数-ldl)
 gcc -Wall -g dy.c -o dy -ldl   #-ldl参数表示连接到动态链接库,不然编译通不过的。
 
阅读(1377) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~