商业化的软件大多支持功能扩展,比较普遍的方法就是使用插件,更实际的形式就是提供一个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上凑集文章,故又翻出张贴于此处,贻笑大方! 
|