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的两种主要链接方式:显式链接和隐式链接。 **一、隐式链接** 隐式链接是Windows程序中最常见的链接DLL的方式。当编译器处理源代码时,它会查找所有引用的外部函数和变量,并将其与DLL中的符号...
本主题主要探讨如何将多个类封装到一个DLL中,并介绍两种调用DLL函数的方式:隐式链接和显式链接。 1. **DLL的封装** 当我们需要在DLL中封装多个类时,首先需要创建这些类的接口,通常是纯虚基类。然后,每个具体...
《DLL中类的显式链接》探讨了在DLL(动态链接库)中如何实现对类的显式链接,这是相对于隐式链接的一种更灵活的技术。显式链接在处理DLL不可用或提供插件功能时尤为实用。下面我们将深入讨论如何进行显式链接,特别...
显式链接是DLL使用的一种方式,与隐式链接相对,它需要程序员在源代码中明确调用LoadLibrary和GetProcAddress等API函数来加载和使用DLL中的函数。下面我们将详细讨论显式链接DLL的实例及其相关知识点。 首先,让...
14.如何显式链接DLL?(Visual C++编程 源代码)14.如何显式链接DLL?(Visual C++编程 源代码)14.如何显式链接DLL?(Visual C++编程 源代码)14.如何显式链接DLL?(Visual C++编程 源代码)14.如何显式链接DLL?...
本文将深入探讨两种调用DLL中函数的方法:隐式调用和显式调用,并通过一个名为`TestCallDll`的示例项目进行解释。 **隐式调用DLL** 隐式调用是通过链接器在编译时完成的。当编译器处理源代码时,如果遇到对DLL中...
- **显式链接**:程序运行时才加载DLL,需要使用LoadLibrary和GetProcAddress等API来动态获取和调用DLL中的函数。这种方法灵活性高,但需要在代码中处理加载和卸载过程。 2. **创建DLL** - 使用C++创建DLL,需要...
本篇文章将深入探讨如何在QT中创建和调用Dll中的方法,特别是类成员方法,我们将按照以下步骤进行详细讲解。 首先,了解DLL的基本概念。DLL(Dynamic Link Library)是Windows操作系统中的一种共享库,它可以被多个...
在显式链接中,你需要使用`LoadLibrary()`和`GetProcAddress()`函数来加载DLL并获取函数指针。例如,假设你的DLL导出了一个名为`MyFunction`的函数: ```cpp // 主程序 HMODULE hDll = LoadLibrary("MyDll.dll"); ...
在显式链接中,程序员需要在代码中明确指定要使用的DLL函数,并在运行时使用LoadLibrary和GetProcAddress这两个API来加载DLL和获取函数指针,然后通过这个指针调用DLL中的函数。这种方式给予开发者更大的控制权,...
总之,动态链接库的显式调用允许开发者在运行时动态地加载和使用DLL中的功能,这对于构建可扩展和灵活的应用程序非常有用。在Qt环境下,可以利用`QLibrary`类来简化这一过程,同时提供良好的错误处理机制。了解和...
1. 创建DLL项目:在VC2008中创建一个新的DLL工程,定义导出函数和类,使用`__declspec(dllexport)`关键字标记。 2. 编写头文件:定义DLL接口,包括导出函数的声明,供客户端程序使用。 3. 创建客户端项目:新建一个...
创建显式链接的DLL与创建隐式链接的DLL过程相似,只需要确保在头文件中正确声明函数即可。 **3.2 使用显式链接的DLL** 使用显式链接的DLL涉及到以下几个步骤: 1. **加载DLL**:使用`LoadLibrary`函数加载DLL。 2...
而在显式链接中,程序员需要手动加载DLL,查找并调用所需的函数,这提供了更大的灵活性,但需要更多的代码来管理。 在描述中提到的“半透明窗口”,可能是指使用DLL实现特定的Windows API功能,如设置窗口的透明度...
在Qt中创建一个显式的DLL(动态链接库)文件,是为了实现代码的重用和模块化,这在软件开发中非常常见。以下将详细介绍如何利用Qt的工具和类,如QLibrary,来创建和使用一个最简单的DLL。 首先,我们需要创建一个...
应用程序中的导入函数与DLL文件中的导出函数进行链接有两种方式:隐式链接和显式链接。一、隐式链接 在建立一个DLL文件时,编译器会自动生成一个与该文件对应的导入库文件(扩展名为lib)。该文件包含了DLL中所有...
1.3.2 显式链接 1.4 导出类 1.4.1 成员类 1.4.2 内联成员函数 1.4.3 友元函数 1.4.4 嵌套类 1.4.5 静态成员变量 1.4.6 查看导出 1.5 导入类 1.5.1 内联成员函数 第 2 章 MFC Regular DLL 2.1 三种 DLL 2.1.1 non-MFC...
与隐式链接不同,显式链接是在运行时通过LoadLibrary和GetProcAddress等API函数动态加载和查找DLL中的函数。在VC6.0中,实现显式链接的过程如下: 1. **创建DLL项目**:与隐式链接相同,创建DLL项目并定义导出函数...
- 显式链接则需使用API函数如`LoadLibrary`和`GetProcAddress`来动态加载和查找DLL函数。 4. **多线程与DLL**: - Delphi DLL支持多线程使用,这意味着DLL中的函数可以在不同的线程中安全调用。 - 但需要注意的...