分享
 
 
 

一个装载DLL的类

王朝vc·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

//原著:Zoran M.Todorovic

//译者:重庆大学光电工程学院 贾旭滨

//欢迎大家批评指教,谢谢!

当你的应用程序要连接到一个DLL的时候,你可以用静态连接或者是动态连接。静态连接比较容易,但是如果你的DLL文件不存在的话,你的应用程序就不能运行了。而动态连接比较难,但却提供了更好的灵活性,如果DLL文件没有找到的话,你可以显示一个对话框来说明要用的DLL文件不存在,但是你的应用程序却由于要使用丢失的DLL文件,而使得有些功能不能用了,但这不会终止你的应用程序的。下面这个类就提供了一个动态加载DLL文件和使用的方法。以下是步骤:

第1步:

类TBaseModule是一个基类。它的子类必须与DLL协调。代码如下:

#define BM_OK 0

#define BM_DLLNOTFOUND 1

#define BM_INVALIDEXPORT 2

class TBaseModule {

protected:

int ErrorCode; // One of BM_xxx defines

int FunctionErrorCode;

CString DLLName;

HINSTANCE DLLHandle;

public:

TBaseModule(CString name);

virtual ~TBaseModule();

virtual BOOL Create(void);

virtual void Destroy(void);

int GetErrorCode(void) { return ErrorCode; }

int GetFunctionErrorCode(void) { return FunctionErrorCode; }

CString& GetDLLName(void) { return DLLName; }

};

第2步:

以下是类定义的一部分。代码如下:

TBaseModule::TBaseModule(CString name)

{

DLLName = name;

DLLHandle = NULL;

ErrorCode = BM_OK;

FunctionErrorCode = BM_OK;

}

TBaseModule::~TBaseModule()

{

Destroy();

}

BOOL TBaseModule::Create(void)

{

DLLHandle = ::LoadLibrary(DLLName);

if (DLLHandle == NULL) {

ErrorCode = BM_DLLNOTFOUND;

return FALSE;

}

return TRUE;

}

void TBaseModule::Destroy(void)

{

if (DLLHandle) {

::FreeLibrary(DLLHandle);

DLLHandle = NULL;

}

}

第3步:

现在,我们可以假设我们要动态的加载一个DLL,并且DLL会输出一些函数,DLL文件的名字是TEST.DLL。

(DLL的调用有个标准的规定):

void ExportedFunction1(int code, const char *str);

WORD ExportedFunction2(DWORD *data);

char *ExportedFunction3(int code);

第4步:

你必须给每个出口函数定义一个类型,在建一个以TBaseModule为基类的类TTestModule。代码如下:

typedef void (__stdcall *FExportedFunction1)(int code, const char *str);

typedef WORD (__stdcall *FExportedFunction2)(DWORD data);

typedef char * (__stdcall FExportedFunction3)(int code);

class TTestModule : public TBaseModule {

private:

FExportedFunction1 FunctionExportedFunction1;

FExportedFunction2 FunctionExportedFunction2;

FExportedFunction3 FunctionExportedFunction3;

public:

TTestModule(CString dllname);

virtual BOOL Create(void);

virtual void Destroy(void);

void Function1(int code, const char *str);

WORD Function2(DWORD data);

char *Function3(int code);

};

第5步:

定义类。代码如下:

TTestModule::TTestModule(CString dllname)

:TBaseModule(dllname)

{

FunctionExportedFunction1 = NULL;

FunctionExportedFunction2 = NULL;

FunctionExportedFunction3 = NULL;

}

BOOL TTestModule::Create(void)

{

if (TBaseModule::Create()) {

FunctionExportedFunction1 = (FExportedFunction1)::GetProcAddress(DLLHandle,_T"ExportedFunction1");

FunctionExportedFunction2 = (FExportedFunction2)::GetProcAddress(DLLHandle,_T"ExportedFunction2");

FunctionExportedFunction3 = (FExportedFunction3)::GetProcAddress(DLLHandle,_T"ExportedFunction3");

if (FunctionExportedFunction1 && FunctionExportedFunction2 && FunctionExportedFunction3)

return TRUE;

ErrorCode = BM_INVALID_EXPORT;

return FALSE;

}

return FALSE;

}

void TTestModule::Destroy(void)

{

TBaseModule::Destroy();

FunctionExportedFunction1 = NULL;

FunctionExportedFunction2 = NULL;

FunctionExportedFunction3 = NULL;

}

void TTestModule::Function1(int code, const char *str)

{

if (DLLHandle == NULL)

TRACE0(_T("TEST.DLL is not loaded - Use Create()\n"));

FunctionErrorCode = BM_OK;

if (FunctionExportedFunction1)

FunctionExportedFunction1(code,str);

else

FunctionErrorCode = BM_INVALIDEXPORT;

}

WORD TTestModule::Function2(DWORD data)

{

if (DLLHandle == NULL)

TRACE0(_T("TEST.DLL is not loaded - Use Create()\n"));

FunctionErrorCode = BM_OK;

if (FunctionExportedFunction2)

return FunctionExportedFunction1(data);

FunctionErrorCode = BM_INVALIDEXPORT;

return 0;

}

char *TTestModule::Function3(int code)

{

if (DLLHandle == NULL)

TRACE0(_T("TEST.DLL is not loaded - Use Create()\n"));

FunctionErrorCode = BM_OK;

if (FunctionExportedFunction3)

return FunctionExportedFunction3(code);

FunctionErrorCode = BM_INVALIDEXPORT;

return NULL;

}

第6步:

接下来你要做的就是声明它。(有几种方法)

1.作为CWinApp的一个数据成员。

2.作为函数内的一个局部对象。

3.作为一个全局变量。

4.作为其它类的数据成员。

不过你必须调用Create()函数,并且返回一个BOOL值,如果是FALSE,你应该检查错误代码,然后决定放弃使用DLL还是使用它。

TTestModule testDll(_T("TEST.DLL"));

... ...

BOOL retcode = testDll.Create();

if (retcode == FALSE) {

if (testDll.GetErrorCode() == BM_INVALIDEXPORT)

::AfxMessageBox(_T("TEST.DLL does not contain all functions"));

else if (testDll.GetErrorCode() == BM_DLLNOTFOUND)

::AfxMessageBox(_T("TEST.DLL is not found"));

}

... ...

testDll.Function1(0,NULL);

if (testDll.GetFunctionErrorCode() == BM_INVALIDEXPORT)

::AfxMessageBox(_T("ExportedFunction1 does not exist in a TEST.DLL"));

... ...

这样就可以了。其他还没有的地方自己加进去就行了。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有