动态链接库

        动态链接库也被叫做为dll,是Dynamic Link Library的缩写。dll是一个被其他应用程序调用的程序模块,其中封装了可以被调用的资源或函数。dll不能够单独运行,它是依附于exe文件创建的进程来执行的。每个程序都可以通过链接dll使用其中包含的接口函数,这有助于避免代码重用和促进内存的有效使用,使程序实现模块化,只在相应的功能被请求时才加载相应模块。

        另外,调用者只需要根据自己要实现的功能调用dll中对应的接口,而不需要知道每个接口函数具体的实现。dll的开发者可以通过dll隐藏接口的代码实现,比如你只想让其他人使用你的算法,但不想让其他人知道你的算法和逻辑代码,就可以把它们封装到dll中。这样,调用人的程序中不会包含算法实际的实现代码,而只有调用dll的接口代码。

目录

动态链接库

建立一个DLL

查看DLL中的函数

调用DLL


建立一个DLL

        打开Visual Studio,新建 -> 项目 -> Visual C++ -> 动态链接库

使用__declspec(dllexport)关键字导出dll

        添加头文件arithmetic.h 在头文件中添加导出函数add函数:

#ifdef __cplusplus         // 一般用于将C++代码以标准C形式输出(即以C的形式被调用)
extern"C"{                 // 告诉编译器下面大括号括起来的函数是c语言函数(因为c++和c语言对函数的编译转换不一样,主要是c++中存在重载)
#endif

__declspec(dllexport) int __stdcall  add(int a, int b);      // __declspec(dllexport)声明导出该函数供外部调用

__declspec(dllexport) int __stdcall  subtruct(int a, int b); // __stdcall是Windows API默认的函数调用协议,函数采用从右到左的压栈方式,自己在退出时清空堆栈,也可以不使用

#ifdef __cplusplus
}
#endif

        __cplusplus是c++中的自定义宏,这个宏表示这是一段C++的代码。 在C++中,为了支持重载机制,在编译生成的汇编码中,要对函数的名字进行一些处理,加入比如函数的返回类型等等.而在C中,只是简单的函数名字而已,不会加入其他的信息.也就是说:C++和C对产生的函数名字的处理是不一样的。这里使用extern "C"就是告诉编译器,以C的方式来链接它括起来的函数。上面的函数的含义就是:如果这是一段C++代码,那么加入extern "C"处理大括号中的代码。

再添加arithmetic.cpp文件实现函数:

#include "Arithmetic.h"

int __stdcall  add(int a, int b)
{
	return a + b;
}

int __stdcall  subtruct(int a, int b)
{
	return a - b;
}

        编写好之后,我们直接右键项目名,点生成就可以项目路径下的debug文件夹里找到我们生成的dll和lib文件


查看DLL中的函数

        有时候,我们需要调用已经写好的dll,但是我们不知道这个dll中有什么函数。我们可以在Visual Studio的安装目录\Common7\Tools\Bin下,找到微软提供的查看已有dll中函数名的工具Dependency Walker

        我们只需要把刚才生成的dll文件拖进Dependency中,就可以看到这个dll中所包含的函数名以及它所依赖的dll


调用DLL

一、静态调用

        1. 我们测试一下调用这个刚才创建的DLLTest的dll,首先创建一个工程,添加如下代码:

#include <iostream>

#pragma comment(lib, "DllTest.lib")                                     // 告诉程序lib文件的路径,这里就表示当前目录
extern "C" __declspec(dllimport) int __stdcall add(int a, int b);       // 这里使用__declspec(dllimport),正好和生成dll时的__declspec(dllexport)对应
extern "C" __declspec(dllimport) int __stdcall subtruct(int a, int b);  // 生成dll时没有加__stdcall的话,这里也不用加

int main()
{
	int a = 3;
	int b = 2;
	std::cout << "add result is: " << add(a, b) << std::endl;
	std::cout << "subtruct result is: " << subtruct(a, b) << std::endl;
}

        要把生成的lib文件放到和工程代码的统一路径下,在然后将dll文件放到和程序要生成的EXE文件同一路径下:

        运行程序,可以看到调用成功:

        2 . 当我们要导入好几个dll,且这些dll中有很多函数时,可以为每个dll新建一个头文件来声明导入的函数,如我们新建一个Arithmetic.h的头文件,把DllTest.dll中所有的函数声明都写在这里:

#ifdef __cplusplus
extern "C" {
#endif

	__declspec(dllimport) int __stdcall add(int a, int b);
	__declspec(dllimport) int __stdcall subtruct(int a, int b);

#ifdef __cplusplus
}
#endif

        然后在所有用到这个dll中函数的cpp文件中包含这个头文件,就可以使用dll中的函数了

#include <iostream>
#include "Arithmetic.h"

int main()
{
	int a = 3;
	int b = 2;
	std::cout << "add result is: " << add(a, b) << std::endl;
	std::cout << "subtruct result is: " << subtruct(a, b) << std::endl;
}

        然后我们还需要添加lib库,可以新建一个名为Lib的文件夹,把所有的dll的lib文都放在里面。然后直接右键项目,选择属性,在属性配置→链接器→常规→附加库目录里添加这个Lib文件夹的路径

        然后在链接器→输入→附加依赖项中添加lib文件的名字,最后一样要把dll文件放在程序要生成的EXE文件同一路径下

        运行程序,可以得到和上面一样的结果。 

二、动态调用

        动态调用不是链接时完成的,而是在运行时完成的,动态调用不会在可执行文件中写入DLL相关的信息,而是直接调用dll中的函数。

// 动态调用DLL库
void DynamicUse()
{
    // 运行时加载DLL库
    HMODULE module = LoadLibraryA("DLLTest1.dll");     // 根据DLL文件名,加载DLL,返回一个模块句柄
    if (module == NULL)
    {
        printf("加载DLLTest1.dll动态库失败\n");
        return;
    }
    typedef int(*AddFunc)(int, int);                  // 定义函数指针类型
    AddFunc add;
    // 导出函数地址
    add = (AddFunc)GetProcAddress(module, "add");     // GetProcAddress返回指向的函数名的函数地址
 
    int sum  = add(100, 200);
    printf("动态调用,sum = %d\n",sum);
}

        一般使用动态调用,都是调用一些系统dll中的几个函数函数时使用这种方法,因为调用我们自己写的dll时,很多函数都会经常用到,动态调用就太麻烦了。

Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐