上一篇文章写了动态库的创建和被应用程序调用的流程。但是在工作中,往往会涉及一个动态库调用另一个动态库的情况,而不是单纯的一个应用程序去调用一个动态库。例如,一个动态库的输入变量和输出变量需经过JSON转换,再跟应用程序发生关联,也就是说JSON转换这一环节充当了中转环节。再如,基于tensorrt的模型部署中,将反推理部分封装成动态库,再提供给另一个检测相关的动态库调用,最后这个动态库再与应用程序产生交互。那么,如何通过一个动态库来调用另一个动态库呢?这篇文章将详细介绍一下此流程。
一、应用程序
这里就不介绍“应用程序”的创建流程了,有兴趣的可以看上一篇文章(【C/C++】基于vs2019创建动态库并实现调用)。还是采用上篇文章的例子,直接上图:
二、动态库创建
1.动态库1
第一个动态库(test_dll.dll),还是采用上篇文章的那个,只是有些地方需要修改一下。第一个地方:在test_dll工程中添加头文件DLL2.h;第二个地方:在dll.cpp中添加#include "DLL2.h"和#pragma comment(lib, "..//test_dll//x64//Release//test_Dll2.lib")。后者还可以在属性里面来实现,操作:属性页-VC++目录-库目录,这里添加..//test_dll//x64//Release;属性页-链接器-输入-附加依赖项,这里添加test_Dll2.lib;第三个地方:print_ok()函数里面的内容改成“print_ok_ok();”。
2.动态库2
动态库创建可以看上一篇文章,这里创建了动态库:test_Dll2.dll,直接上图:
三、运行结果
四、附加代码
test.cpp
#pragma once
#include <iostream>
#include <Windows.h>
using namespace std;
typedef void(*TestDll)();
class Test
{
public:
Test(void);
~Test(void);
int testDll();
private:
HINSTANCE hdll;
TestDll testdll;
};
Test::Test(void)
{
hdll = LoadLibraryA("test_dll.dll");
if (hdll == NULL)
{
FreeLibrary(hdll);
return;
}
testdll= (TestDll)GetProcAddress(hdll, "print_ok");
if (testdll == NULL)
{
FreeLibrary(hdll);
return;
}
}
int Test::testDll()
{
testdll();
return 0;
}
Test::~Test(void)
{
if (hdll)
FreeLibrary(hdll);
}
static Test TD;
int main()
{
int one=TD.testDll();
getchar();
return 0;
}
dll.h
#pragma once
#ifdef DLL_EXPORTS
#define DLL_API __declspec(dllexport)
#else
#define DLL_API __declspec(dllimport)
#endif
#include <fstream>
extern "C"
{
DLL_API void print_ok();
};
dll.cpp
#include "pch.h"
#include "dll.h"
#include "DLL2.h"
#include <iostream>
using namespace std;
#pragma comment(lib, "..//test_dll//x64//Release//test_Dll2.lib")
DLL_API void print_ok()
{
print_ok_ok();
}
DLL2.h
#pragma once
#ifdef DLL2_EXPORTS
#define DLL2_API __declspec(dllexport)
#else
#define DL2L_API __declspec(dllimport)
#endif
#include <fstream>
extern "C"
{
DLL2_API void print_ok();
};
DLL2.cpp
#include "pch.h"
#include "DLL2.h"
#include <iostream>
using namespace std;
DLL2_API void print_ok_ok()
{
std::cout << "ok_ok123456789" << endl;
}