win32 dll简单例子

来源:互联网 发布:苹果手机价格评估软件 编辑:程序博客网 时间:2024/04/25 12:10

一。显示链接dll

编写dll

FILE->Visal C++项目: Win32项目->应用程序设置: 选择 DLL(D) 选项 并勾选 导出符号,将 h,cpp文件修改如下:

MyDll.h

   //Mydll.h
   #ifdef MYDLL_EXPORTS
#define MYDLL_API __declspec(dllexport)
#
else
#define MYDLL_API __declspec(dllimport)
#endif

extern 
"C" MYDLL_API int fun(int mode); //自己写的 extern "C" 不可少
extern "C" MYDLL_API int fun2(int a,int b); 

MyDll.cpp

 

#include "stdafx.h"
#include 
"MyDll.h"
BOOL APIENTRY DllMain( HANDLE hModule, 
                       DWORD  ul_reason_for_call, 
                       LPVOID lpReserved
                     )
{
    
switch (ul_reason_for_call)
    
{
    
case DLL_PROCESS_ATTACH:
    
case DLL_THREAD_ATTACH:
    
case DLL_THREAD_DETACH:
    
case DLL_PROCESS_DETACH:
        
break;
    }

    
return TRUE;
}


MYDLL_API 
int fun(int mode) //自己写的
{
    
    
return mode*mode;
}


MYDLL_API 
int fun2(int a,int b) //自己写的
{
    
int d = (a>b?(a-b):(b-a));
    
return d;
}

编写测试程序:testDll

采用win32控制台生成的执行程序进行测试 (注: 属性->C/C++:预处理器->预处理器定义  加宏:MYDLL_EXPORTS)
因为MyDll.h中定义了宏 #define MYDLL_API __declspec(dllexport)

 

#include <iostream>
#include 
<Windows.h>

typedef 
int (*PFNMYDLL)(int);//声明函数原型
typedef int (*HHH)(int,int); 

using namespace std;

void main()
{
    HMODULE hModule 
= ::LoadLibrary("MyDll.dll");//加载DLL库

    PFNMYDLL newfun 
= (PFNMYDLL)::GetProcAddress(hModule,"fun");//取得fun函数的地址

    
int i = newfun(4);
    printf(
"The result is %d ",i);

    HHH newfun2 
= (HHH)::GetProcAddress(hModule,"fun2");//取得fun函数的地址

    
int d = newfun2(6,4);
    printf(
"the 6,4 is: %d ",d);

    
int c = newfun2(7,19);
    printf(
"the 7,19 is:%d ",c);

    ::FreeLibrary(hModule);
}

 

 

二.隐式链接

[cpp] view plaincopy
  1. #ifdef MYDLL_EXPORTS  
  2. #define MYDLL_API __declspec(dllexport)  
  3. #else  
  4. #define MYDLL_API __declspec(dllimport)  
  5. #endif  
  6.   
  7. class MYDLL_API MyDll  
  8. {  
  9. public:  
  10.     MyDll(void);  
  11.     ~MyDll(void);  
  12.     void setValue(int value);  
  13.     int getValue();  
  14.   
  15. private:  
  16.     int m_nValue;  
  17. };  

 

 使用 dll 代码

[cpp] view plaincopy
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <windows.h>  
  4. #include "MyDll.h"  
  5. #pragma comment(lib,"MyDll.lib")  
  6.   
  7. void main()  
  8. {  
  9.   
  10.  MyDll myDll;  
  11.  myDll.setValue(20);  
  12.  int i = myDll.getValue();  
  13.   
  14.  printf("%d",i);  
  15. }  

 

 以下为转贴

三。导出并显式链接一组C++成员函数

这里有两个问题。第一是C++成员函数名是经过修饰的(即使指定extern "C"标记也是这样);第二是C++不允许将指向成员函数的指针转换成其它类型。这两个问题限制了C++类的显式链接。下面介绍两种方法来解决这个问题:①用虚函数表的方法,这也是COM使用的方法;②用GetProcAddress直接调用。

 

1.虚函数表方法:

使用到的 dll 头文件 MyDll.h

[cpp] view plaincopy
  1. #ifdef MYDLL_EXPORTS  
  2. #define MYDLL_API __declspec(dllexport)  
  3. #else  
  4. #define MYDLL_API __declspec(dllimport)  
  5. #endif  
  6.   
  7. class MYDLL_API MyDll  
  8. {  
  9. public:  
  10.     MyDll(void);  
  11.     MyDll(int i);  
  12.     virtual ~MyDll(void);  
  13.     virtual void setValue(int value);  
  14.     virtual int getValue();  
  15.   
  16. private:  
  17.     int m_nValue;  
  18. };  

 

使用 dll 的代码

[cpp] view plaincopy
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <string>  
  4. #include <windows.h>  
  5.   
  6. #include "MyDll.h"  
  7.   
  8. typedef MyDll* (*pCreateA)();  
  9. typedef MyDll* (*pCreateA1)(int);  
  10.   
  11. void main()  
  12. {  
  13.     HMODULE hModule;  
  14.       
  15.     hModule = ::LoadLibrary("MyDll");//加载DLL库  
  16.   
  17.     pCreateA pCreate = (pCreateA)GetProcAddress(hModule, TEXT("CreateMyDll"));  
  18.   
  19.     MyDll* a = (pCreate)();  
  20.     a->setValue(20);  
  21.     printf("one:%d/n",a->getValue());  
  22.   
  23.   
  24.   
  25.     pCreateA1 pCreate1 = (pCreateA1)GetProcAddress(hModule, TEXT("CreateMyDll1"));  
  26.   
  27.     MyDll* b = (pCreate1)(50);  
  28.     printf("two:%d/n",b->getValue());  
  29.   
  30.     ::FreeLibrary(hModule);  
  31.   
  32.     getchar();  
  33.     return;  
  34. }  

 

dll 项目

MyDll.h 即使用到的 dll 头文件

MyDll.cpp

[cpp] view plaincopy
  1. #include "MyDll.h"  
  2.   
  3. MyDll::MyDll(void)  
  4. :m_nValue(0)  
  5. {  
  6. }  
  7.   
  8. MyDll::MyDll(int i)  
  9. {  
  10.     m_nValue = i;  
  11. }  
  12.   
  13. MyDll::~MyDll(void)  
  14. {  
  15.     m_nValue = 0;  
  16. }  
  17.   
  18. void MyDll::setValue(int value)  
  19. {  
  20.     m_nValue = value;  
  21. }  
  22.   
  23. int MyDll::getValue()  
  24. {  
  25.     return m_nValue;  
  26. }  

 

Inst.cpp

[c-sharp] view plaincopy
  1. #include "MyDll.h"  
  2.   
  3. extern "C" __declspec(dllexport) MyDll* CreateMyDll()  
  4. {  
  5.     return new MyDll();  
  6. }  
  7. extern "C" __declspec(dllexport) MyDll* CreateMyDll1(int i)  
  8. {  
  9.     return new MyDll(i);  
  10. }  

 

 

 

这个方法的使用得用户可以很容易地为你的程序制作插件。它的缺点是创建对象的内存必须在dll中分配.

 

直接使用GetProcAddress进行显式链接

这个方法的关键在于将GetProcAddress函数返回的FARPROC类型转化为C++中指向成员函数的指针。幸运的是,通过C++的unio和模板机制,这个目标可以很容易地实现。我们要做的只是定义如下的函数:

template<class Src , class Dest>

Dest force_cast(Src src){

 union{

  Dest d;

  Src s;

 } convertor;

convertor.s = Src;

 return convertor.d;

}

上面的函数允许我们在任何类型间进行转换,比reinterpret_cast更加有效。例如,我们定义一种指针类型:

typedef void (A::*PSetNum)(int);

我们可以将FARPROC类型的指针fp转化成PSetNum:

PSetNum psn = force_cast<PSetNum>(fp);

找到了将FARPROC转化成成员函数指针的方法以后,我们要考虑如何将C++成员函数以更加友好的名字导出。这可以通过一个.def文件来实现。

第一步是找到待导出函数经过修饰的函数名,这可以通过查看map file或者汇编代码来实现。然后在.def文件中指定导出函数的新的函数名:

EXPORTS

 ConstructorOfA1 = ??0A@@QAE@XZ        PRIVATE

 ConstructorOfA2 = ??0A@@QAE@H@Z       PRIVATE

 SetValueOfA       = ?SetNum@A@@UAEXH@Z  PRIVATE

 GetValueOfA       = ?GetNum@A@@UAEHXZ   PRIVATE  

 

下面是调用这些成员函数的方法:

typedef void (A::*PfnConstructorOfA1)();

typedef void (A::*PfnConstructorOfA2)(int);

typedef void (A::*PfnDestructorOfA)();

typedef void (A::*PfnSetNumOfA)(int);

typedef int  (A::*PfnGetNumOfA)();

A* a1 = (A*)_alloca(sizeof(A));

 

PfnConstructorOfA1 pfnConsA =

     force_cast<PfnConstructorOfA1>(GetProcAddress(hMod, TEXT("ConstructorOfA1")));

(a1->*pfnConsA)();

PfnSetNumOfA pfnSetNumA =

          force_cast<PfnSetNumOfA>(GetProcAddress(hMod, TEXT("SetNumOfA")));

(a1->*pfnSetNumA)(1);

           

PfnGetNumOfA pfnGetNumA =

          force_cast<PfnGetNumOfA>(GetProcAddress(hMod, TEXT("GetNumOfA")));

_tprintf(TEXT("Value of m_nNum in a is %d/n"),(a1->*pfnGetNumA)());

 

 注意这里使用了alloca从栈中分配内存,你也可以使用malloc从堆中分配内存。但是不能使用C++的new操作符,因为能过new来分配内存编译器会自动插入对constructor的调用。但我们要的是显式链接,所以必须避免这种情况。随之产生的结果是我们只能显式地去调用构造函数和析构函数。

原文出自:

http://www.codeguru.com/cpp/w-p/dll/importexportissues/article.php/c123/


转自:http://blog.csdn.net/rem2002/article/details/1744978

0 0
原创粉丝点击