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所指示的动态链接库; |
编译函数源程序时选用-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参数表示连接到动态链接库,不然编译通不过的。