DLL的显式链接在某些时候比隐式链接具有更大的灵活性。比如,如果在运行时发现DLL无法找到,程序可以显示一个错误信息并能继续运行。当你想为你的程序提供插件服务时,显式链接也很有用处。

显式链接到全局C/C++函数非常简单。假设你想调用DLL中的一个函数ExportedFn,你可以像这样很简单地导出它:

extern "C" _declspec(dllexport)
void ExportedFn(int Param1, char* param2);

必须使用extern "C"链接标记,否则C++编译器会产生一个修饰过的函数名,这样导出函数的名字将不再是ExportedFn,而是一个形如"??ExportedFn@QAEX”的名字。假设这个函数从DLL1.dll导出,那么客户端可以像这样调用这个函数:

HMODULE hMod = LoadLibrary("Dll1.dll");
typedef void (*PExportedFn)(int, char*);
PExportedFn pfnEF = (PExportedFn)GetProcAdress("ExportedFn");
pfnEF(1, "SomeString");

如果你想导出并显式链接一组C++成员函数又该怎么办呢?这里有两个问题。第一是C++成员函数名是经过修饰的(即使指定extern "C"标记也是这样);第二是C++不允许将指向成员函数的指针转换成其它类型。这两个问题限制了C++类的显式链接。下面介绍两种方法来解决这个问题:①用虚函数表的方法,这也是COM使用的方法;②用GetProcAddress直接调用。我将以下面这个类为例进行讲解:

class A
{private:int m_nNum;
public:  A();A(int n);virtual ~A();void SetNum(int n);int GetNum();
};

一.用虚函数表进行显式链接

这个方法是COM的基础。当我们定义一组虚函数的时候,编译器会创建一个虚函数表,将各虚函数的地址按声明的顺序放入其中。当一个类对象被创建时,它的前四个字节是一个指针,指向这个虚函数表。如果我们将A的定义修改成这样:

class A
{
private:int m_nNum;
public:        A();A(int n);virtual ~A();virtual void SetNum(int n);virtual int GetNum();
};

那么一个虚函数表将被编译器创建出来,其中包含三个函数的地址:析构函数,SetNum和GetNum。现在类对象要在dll中创建。既然我们要显式链接,就需要一些全局导出函数来调用operator new以创建对象。因为A有两种构造函数,所以我们定义两个函数CreateObjectofA()和CreateObjectofA1(int)并将其导出。客户可以这样来使用类对象:

typedef A* (*PFNCreateA1)();
PFNCreateA1 pfnCreateA1 =  (PFNCreateA1)GetProcAddress(hMod, TEXT("CreateObjectofA1"));
A* a = (pfnCreateA1)();
a->SetNum(1);_tprintf(TEXT("Value of m_nNum in a is %d\n"),a->GetNum());
delete a;
要注意的是CreateObjectofA必须使用operator new来创建对象这样客户端才可以安全地调用operator delete来销毁对象:
extern "C" __declspec(dllexport) A* CreateObjectofA1()
{return new A();
}

这个方法的使用得用户可以很容易地为你的程序制作插件。它的缺点是创建对象的内存必须在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
SetNumOfA = ?SetNum@A@@UAEXH@Z PRIVATE
GetNumOfA = ?GetNum@A@@UAEHXZ PRIVATE
DestructorOfA = ??1A@@UAE@XZ 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)());PfnDestructorOfA pfnDestA = force_cast<PfnDestructorOfA>(GetProcAddress(hMod, TEXT("DestructorOfA")));
(a1->*pfnDestA)();

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

转:http://www.moon-soft.com/doc/14639.htm

【转】DLL中类的显式链接相关推荐

  1. vc++ 显式链接dll

    Visual C++ 显式链接 在显式链接下,应用程序必须进行函数调用以在运行时显式加载 DLL.为显式链接到 DLL,应用程序必须: 调用 LoadLibrary(或相似的函数)以加载 DLL 和获 ...

  2. 关于Linux中so显式链接(dlopen)找不到函数符号地址的问题

    问题背景 在做项目的时候,遇到一个so调用问题,既别人提供了一些so库,其中一个so库包含了给我调用的函数,而这个库里面的函数又调用了其他库的函数,这些所有的库都是linux下编译出来的,而项目则是需 ...

  3. vc++ 隐式链接dll

    Visual C++ 隐式链接 为隐式链接到 DLL,可执行文件必须从 DLL 的提供程序获取下列各项: 包含导出函数和/或 C++ 类的声明的头文件(.h 文件).类.函数和数据均应具有 __dec ...

  4. 使用extern C改善显式调用dll

    extern "C"的简单解析         我们前面介绍了显式调用dll的方法,例如  http://www.cnblogs.com/laogao/archive/2012 ...

  5. C++ dll的隐式与显式调用

     转载自:http://blog.sina.com.cn/s/blog_53004b4901009h3b.html 应用程序使用DLL可以采用两种方式:一种是隐式链接,另一种是显式链接.在使用D ...

  6. 4.1.4 OS之文件的物理结构(连续分配、链接分配[隐式-显式]、索引分配[链接方案-多层索引-混合索引])

    文章目录 0.思维导图 1.文件块.磁盘块 2.连续分配 3.链接分配 隐式链接 显式链接 链接分配总结 4.索引分配 链接方案 多层索引 混合索引 索引分配总结 5.文件物理结构分配总结 0.思维导 ...

  7. 操作系统之文件管理:5、文件物理结构(连续分配、链式(显式、隐式)分配、索引分配(链接、多层索引、混合索引))

    3.文件物理结构 思维导图 文件块.磁盘块 文件分配方式 1.连续分配 2.链接分配 隐式链接 显式链接 3.索引分配 如果一个文件的大小超过一个磁盘块怎么办? 1.链接方案 2.多层索引 3.混合索 ...

  8. 一文彻底搞懂静态库和动态库,显示链接和隐式链接

    定义:运行时库 静态库 动态库 运行时库:Unix中一个典型的运行时库例子就是libc,它包含标准的C函数,如,print(),exit()等等,用户能创建他们自己的运行库(在Windows中是DLL ...

  9. Qt DLL总结【一】-链接库预备知识

    目录 Qt DLL总结[一]-链接库预备知识 Qt DLL总结[二]-创建及调用QT的 DLL Qt DLL总结[三]-VS2008+Qt 使用QPluginLoader访问DLL 1.链接库概念 静 ...

最新文章

  1. NameError: name xx is not defined
  2. 微擎获取openid_微擎中使用微信之门接口,让订阅号也能直接以网页的方式获取OpenID...
  3. 前端学习(575):margin无效情形之鞭长莫及导致无效
  4. js实现select跳转
  5. Mysql基础系列(一)
  6. VS2012错误之 warning LNK4075: 忽略“/EDITANDCONTINUE”(由于“/SAFESEH”规范)
  7. r语言 生成等差序列_使用序列模型生成自然语言
  8. [Threejs]环境光与HDR贴图
  9. win98 支持html5,win98支持1G以上内存的解决办法
  10. PHP 生成 ppt,PHP导出PPT方法,PowerPoint/PhpPresentation处理
  11. 车牌识别存储云服务器,云端(服务器)车牌识别技术
  12. excel使用教程_火遍全球的14个Excel学习网:大神套路、视频课、软件下载应有尽有...
  13. 有没有中文域名SSL证书?如何申请
  14. MATLAB 神经网络NAR时间序列做预测
  15. python dot_graphviz,dot,及dot图可视化
  16. 如何在Win10不同设备之间同步便签
  17. 生产力高于一切?我们为什么会对技术上瘾?
  18. PG数据库内核分析学习笔记_XLOG日志恢复策略
  19. 服务端开发框架及技术(转)
  20. 【JavaScript】获取指定字符串

热门文章

  1. Celery异步调度框架(二)与Django结合使用
  2. linux系统编程之(一) 信号量
  3. 进程与线程的区别?--多线程与线程池
  4. BAT文件执行完成后如何删除自身的解决办法
  5. CoreMotion(加速计)
  6. 把view放在地图覆盖物上
  7. 使用SqlDependency监测数据库
  8. 文本分类(一)封装分词器
  9. 【python】人机大战
  10. 虚拟机服务器被攻击,Linux服务器被攻击用来挖矿了