关于软件中的插件类功能的一个实现

[入库:2005年8月19日] [更新:2007年3月24日]

本文简介:选择自 remix 的 blog

商业化的软件大多支持功能扩展,比较普遍的方法就是使用插件,更实际的形式就是提供一个dll或类似的文件,由商业软件在载入时自动检测这些dll文件,来实现功能的扩展,在公司的某个产品开发中,我也使用了类似的技术,但我使用的方法并没有参考网上已有的代码,而是自己构思了一种方法,在原理上可能与此技术也相似。(我并没有真正的去研究那些插件实现的方法,呵呵,所以无法确认是否一致)

主要原理就是商业软件在载入内存运行时,会自动检测当前目录下的某个目录夹,例如extra目录,枚举所有的dll文件(如果为了保密,可将dll后缀改为其它名称,甚至可将此文件进行加密,在内存中解密后再进行调用,但这不在本文讨论范围内),并根据预定义的函数名进行功能调用(注意:里面很重要的一点,就是所有的dll文件提供的函数接口名都必须相同),我在此定义了2个函数接口,一是getinfo函数,主要返回此扩展模块的版本号及一些相关信息,二是dowork函数,也就是真正的需要被我们调用的工作函数,在此,我们可以设想到,我们完全可以由dowork函数作为入口,在此函数体内实现我们所有需要的功能,例如画一个对话框,或其它一些我们想做的事,由此商业软件的功能将由这些扩展模块进行无限扩展。


术语重点:win32 dll编制接口规范/win32 dll调用规范
(具体规范要求,可以参考windows深入编程类书籍)
   

开发工具:visual c++ 6.0(sp5)

 

以下列出了我在相关实现中的部分代码:
---------------------------------------------------------
扩展模块函数定义模块:(dll中被导出的2个函数定义,可被其它dll/exe文件调用)
public:
bool getinfo(char * myinfo_data);    //返回dll版本号及其它一些信息,结果将保存至myinfo中,可将此char *强制转换成一自定义struct
long dowork(char * name,int usenum,char * work_para);//实际工作函数


商业软件中调用dll模块部分实现:
//检索所有的扩展dll库
hinstance hdll;               // 函数dll句柄
char* myinfo_tmp;
myinfo info;   //我自定义的一个struct的引用
myinfo_tmp=(char*)malloc(sizeof(myinfo));//分配内存

typedef bool (callback* lpfndllfunc1)(char *); //回叫函数声明
lpfndllfunc1 getinfo;    // 函数指针
lpfndllfunc1 dowork;    // 函数指针

int pst=0;  //扩展dll函数个数计数
cstring path,filename,othermsg;
win32_find_data wfd; //mfc中的一个struct引用
path.format("extra\\*.dll");
handle handle=findfirstfile(path,&wfd);  //打开枚举句柄
//以下是枚举extra目录下所有后缀为dll的代码
if(handle!=invalid_handle_value) 
{
 do
 {
  filename.format("extra\\%s",(cstring)wfd.cfilename);
  hdll = loadlibrary(filename); //载入检索到的dll文件到内存
  if (hdll != null)
  {
   getinfo = (lpfndllfunc1)getprocaddress(hdll, "getinfo"); //取得getinfo函数在内存中的入口地址
   if(getinfo)
   {
    getinfo(myinfo_tmp);
    memcpy(&info,myinfo_tmp,sizeof(myinfo));
    m_list.addstring(info.name); //将name加入list控件中
    othermsg.format("厂商:%s\n\n文件名:%s\t版本号:%s\n\n其它信息:%s",info.name,info.dllname,info.ver,fwinfo.other);

    dowork=(lpfndllfunc2)getprocaddress(hdll,"dowork");    //取得dowork函数在内存中的入口地址

    if(!dowork("name",3,"now"))         afxmessagebox("调用dowork函数时出错!");
   }
   pst++;
  }
 }
 freelibrary(hdll); //释放句柄
}
while(findnextfile(handle,&wfd));
 
findclose(handle); //关闭枚举句柄
free(myinfo_tmp); //释放内存

---------------------------------------------------------

以上是相关调用的实现代码,大家如有疑问,可发邮件于我,或在此留言,呵呵,可能会有更巧妙的方法实现扩展功能,希望我能抛砖引玉,引出更好的实现方法来。  

注:此文为原创,写于2001年,为在此blog上凑集文章,故又翻出张贴于此处,贻笑大方!

本文关键:关于软件中的插件类功能的一个实现
  相关方案
Google
 

本站最佳浏览方式为 分辨率 1024x768 IE 6.0(或更高版本的 IE浏览器)

go top