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