C++_编写动态链接库

原文:http://blog.csdn.net/a7055117a/article/details/47733247

动态链接库简介

动态链接库(Dynamic Link Library 或者 Dynamic-link Library,缩写为 DLL),是微软公司在微软Windows操作系统中,实现共享函数库概念的一种方式。这些库函数的扩展名是 ”.dll”、”.ocx”(包含ActiveX控制的库)或者 “.drv”(旧式的系统驱动程序)。 
动态链接提供了一种方法,使进程可以调用不属于其可执行代码的函数。函数的可执行代码位于一个 DLL 文件中,该 DLL 包含一个或多个已被编译、链接并与使用它们的进程分开存储的函数。DLL 还有助于共享数据和资源。多个应用程序可同时访问内存中单个 DLL 副本的内容。 
通过使用DLL可以实现模块化,使之由相对独立的组件组成,可以更快的加载应用各个模块的功能,还可以更容易的将更新应用于各个模块。 
例如,一个程序含有很多个功能,将他们都编写成DLL文件,当软件更新时,只需更新相应的DLL文件而不需要重新安装整个程序

编写一个DLL

1、新建工程,使用VS2015编写一个实现两数相加的DLL 
文件->新建项目->win32项目->下一步->DLL->勾选预编译头文件,导出符号->完成

此时VS2015会自动生成3个头文件和3个cpp文件 
这里写图片描述

其中TestDll.h是DLL的头文件

#ifdef TESTDLL_EXPORTS
#define TESTDLL_API __declspec(dllexport)
#else #define TESTDLL_API __declspec(dllimport) #endif // 此类是从 TestDll.dll 导出的 class TESTDLL_API CTestDll { public: CTestDll(void); // TODO: 在此添加您的方法。 }; //导出的变量 extern TESTDLL_API int nTestDll; //导出函数 TESTDLL_API int fnTestDll(void);

stdafx.h包含需要的头文件

#pragma once
#include "targetver.h"
#define WIN32_LEAN_AND_MEAN             
#include <windows.h>

TestDll.cpp是源文件 
DllMain是动态链接库的入口点,在DllMain函数中,hModule参数是该DLL模块的句柄,代表这个文件的镜像加载到进程的地址空间使用的基地址,ul_reason_for_call参数的指标是本次调用的原因,包括

  • DLL_PROCESS_ATTACH,表示动态链接库刚被某一个进程加载,映射到了某一个地址空间,可以在此进行初始化
  • DLL_THREAD_ATTACH,动态链接库将被卸载,可以在这进行资源释放
  • DLL_THREAD_DETACH,应用程序创建了一个新的进程
  • DLL_PROCESS_DETACH,某个进程正常终止
// dllmain.cpp : 定义 DLL 应用程序的入口点。
#include "stdafx.h"

BOOL APIENTRY DllMain( HMODULE 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; }

2、将TestDll.h中的导出函数改为

TESTDLL_API int fnTestDll(int a,int b);

删除导出变量和导出类部分

3、将TestDll.cpp文件中导出函数改为

TESTDLL_API int fnTestDll(int a, int b)
{
    return a+b;
}

删除导出类和导出变量部分 
注:Dll能够定义两种函数,一种是内部函数,一种是导出函数。内部函数只能被定义这个函数的模块使用,而导出函数不仅可以在本模块调用,还可以被其他模块调用。dll的主要功能就是向外导出函数供其他模块使用

此时一个简单的两数相加的DLL就写完了,接下来便是如何加载这个Dll了,加载这个dll有两种方法,一种是隐式动态链接,一种是显示动态链接

隐式动态链接

新建一个win32控制台程序,生成刚刚写的dll,将TestDll.lib,TestDll.dll,TestDll.h复制到该控制台程序的文件夹下,然后添加下面的代码

#include "stdafx.h"
#include <iostream>
#include "TestDll.h"

using namespace std; #pragma comment(lib,"TestDll") int main() { int a, b, result; while (cin >> a >> b) { result = fnTestDll(a, b); cout << "Result : " << result << endl; } return 0; }

#pragma comment(lib,”TestDll”)语句表示要连接到TestDll.lib库,使用隐式动态链接需要指明库所在的位置(本例中在当前文件夹下),静态变异也会使可执行文件的体积变大 
测试结果 
这里写图片描述 
可以看到运行成功

显式动态链接

模块使用LoadLIbrary()或者LoadLIbraryEx()函数显式加载DLL,加载后调用GetProcAddress()函数取得DLL导出函数的地址,然后调用函数。

1、在原DLL工程中建立一个模块定义文件(DEF)来指定要导入的函数。 
在TestDll中项目->添加新项->模块定义文件(.def) 
然后在其中添加

EXPORTS
fnTestDll

fnTestDll表示要向外导出的函数名 
重新生成该项目 
2、新建一个win32控制台程序,添加如下代码

// ExplicitTestDll.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <iostream>
#include <Windows.h> using namespace std; typedef int(*PFNEXPORTFUNC)(int, int); int main() { int a, b, result; while (cin >> a >> b) { HMODULE hModule = LoadLibrary(_T("TestDll.dll")); if (hModule != NULL) { PFNEXPORTFUNC mDLLFuncAdd = (PFNEXPORTFUNC)GetProcAddress(hModule, "fnTestDll"); if (mDLLFuncAdd != NULL) { result = mDLLFuncAdd(a, b); cout << "Result: " << result << endl; } FreeLibrary(hModule); } } return 0; }

显式调用DLL分为三个步骤

  1. 声明要导出的DLL函数
  2. 加载目标DLL,即 LoadLibrary()函数,将DLL加载到进程的虚拟地址空间,若成功则返回该DLL模块的句柄,否则返回NULL
  3. 获得导出函数的地址,即GetProcAddress()函数,成功时返回函数地址,否则返回NULL

测试结果 
这里写图片描述

转载于:https://www.cnblogs.com/zhaiyf/p/8548963.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值