参考资料:



 

生成Dll三步走

第一步:先建一个Dll项目

New --> Project --> Dynamic-Link Library(DLL) --> 取名,选路径 --> OK

第二步:编写头文件,例子是一个四则运算

selfTrainingDll.h

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

#pragma once
#ifdef DLL_TRAINING_API
#else                                                                            
#define DLL_TRAINING_API _declspec(dllimport) //当编译时,头文件不参加编译,所以.cpp文件中先定义,后头文件被包含进来,因此外部使用时,为dllexport,而在内部编译时,则为dllimport
#endif  

class DLL_TRAINING_API arithmetic_operation              //需要被外界调用的类(父类)
{
public:
    double Add(double a, double b);
    double Sub(double a, double b);
    double Multi(double a, double b);
    double Div(double a, double b);
};

int DLL_TRAINING_API export333();

View Code

 

第三步:编写CPP文件,实现方法

selfTrainingDll.cpp

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

// selfTrainingDll.cpp : Defines the exported functions for the DLL application.
//

#include "stdafx.h"

#define DLL_TRAINING_API _declspec(dllexport)

#include <iostream>
#include "selfTrainingDll.h"
using namespace std;

double DLL_TRAINING_API arithmetic_operation::Add(double a, double b) {
    return a+b;
}

double DLL_TRAINING_API arithmetic_operation::Sub(double a, double b) {
    return a - b;
}

double DLL_TRAINING_API arithmetic_operation::Multi(double a, double b) {
    return a * b;
}

double DLL_TRAINING_API arithmetic_operation::Div(double a, double b) {
    return a / b;
}

int DLL_TRAINING_API export333() {
    return 333;
}

View Code

 

第四步:生成Dll

Build --> Build Solution

至此,文件生成完毕

 

静态方法调用Dll文件

第一步:创建一个控制台程序

省略

第二步:编译运行,产生Debug文件夹

第三步:将之前Dll项目生成的selfTrainingDll.h和selfTrainingDll.lib放入项目文件夹下,将selfTrainingDll.dll放入Debug文件夹下

第四步:在项目中添加selfTrainingDll.h头文件

第五步:在Cpp中调用Dll

UseSelfDll.cpp

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

// UseSelfDll.cpp : This file contains the 'main' function. Program execution begins and ends there.
//

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

using namespace std;

#include "selfTrainingDll.h"
#pragma comment(lib,"selfTrainingDll.lib")

int main()
{
    arithmetic_operation ao;
    cout << ao.Add(1,2) << endl;
    cout << ao.Sub(2,1) << endl;
    cout << ao.Multi(2,1) << endl;
    cout << ao.Div(6,4) << endl;
    cout << export333() << endl;
    cout << "Hello World!\n"; 
}

View Code

 

至此,调用成功

 

动态方法调用Dll文件

UseSelfDll.cpp

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

// UseSelfDll.cpp : This file contains the 'main' function. Program execution begins and ends there.
//

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

using namespace std;

//#include "selfTrainingDll.h"
//#pragma comment(lib,"selfTrainingDll.lib")

#include <windows.h>

int main()
{

    typedef int(*_print)();
    cout << "1" << endl;
    HINSTANCE hDll = LoadLibrary(L"selfTrainingDll.dll");
    cout << "2" << endl;
    _print pAdd = (_print)GetProcAddress(hDll, (LPCSTR)MAKEINTRESOURCE(7));
    cout << "3" << endl;
    int a = pAdd();
    cout << a << endl;
    //arithmetic_operation ao;
    //cout << ao.Add(1,2) << endl;
    //cout << ao.Sub(2,1) << endl;
    //cout << ao.Multi(2,1) << endl;
    //cout << ao.Div(6,4) << endl;
    //cout << export333() << endl;
    cout << "Hello World!\n"; 
    FreeLibrary(hDll);
}

View Code

由于C++导出Dll时会出现名字更改的问题,因此这里用序列号代表函数,至于函数的序列号可以用如下方法查看:

用VS打开cmd窗口(Tools --> Visual Studio Command Prompt),运行dumpbin -exports xxx.dll     后面最好写DLL的绝对路径,否则可能会报错LNK1181: cannot open input file 'XXX.dll'。

可在EXE所在的目录下使用dumpbin -imports xxx.EXE来查看某EXE文件使用过哪些dll库.

64位EXE尽量去调用64位DLL,同理32位尽量调用32位。

如果想要直接使用函数名,那么在生成DLL时要加extern "C"

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

#pragma once
#ifdef __cplusplus               // if used by C++ code
extern "C" {                     // we need to export the C interface
#endif

#ifdef DLL_TRAINING_API
#else                                                                            
#define DLL_TRAINING_API _declspec(dllimport) //当编译时,头文件不参加编译,所以.cpp文件中先定义,后头文件被包含进来,因此外部使用时,为dllexport,而在内部编译时,则为dllimport
#endif  

    class DLL_TRAINING_API arithmetic_operation              //需要被外界调用的类(父类)
    {
    public:
        double Add(double a, double b);
        double Sub(double a, double b);
        double Multi(double a, double b);
        double Div(double a, double b);
    };

    int DLL_TRAINING_API export333();

#ifdef __cplusplus
}
#endif

View Code

 

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

// UseSelfDll.cpp : This file contains the 'main' function. Program execution begins and ends there.
//

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

using namespace std;

//#include "selfTrainingDll.h"
//#pragma comment(lib,"selfTrainingDll.lib")

#include <windows.h>

int main()
{
    typedef int (*_print)();
    HINSTANCE hDll = LoadLibrary(L"selfTrainingDll.dll");
    _print pAdd = (_print)GetProcAddress(hDll, "export333");
    int a = pAdd();
    cout << a << endl;
    //arithmetic_operation ao;
    //cout << ao.Add(1,2) << endl;
    //cout << ao.Sub(2,1) << endl;
    //cout << ao.Multi(2,1) << endl;
    //cout << ao.Div(6,4) << endl;
    //cout << export333() << endl;
    cout << "Hello World!\n"; 
    FreeLibrary(hDll);
}

View Code

 

但是它仍然有局限性,只针对函数,不针对类。 

 

另外对于带参数的函数,怎么动态调用呢。

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

HINSTANCE hDLL;
typedef DWORD(*GetPCICFG)(BYTE bBus, BYTE bDev, BYTE bFun, BYTE bIdx, PDWORD pdwPortVal, BYTE bSize);
GetPCICFG getPCICFG;
hDLL = LoadLibrary(L"PCI.dll");
if (hDLL == NULL)
    printf("Error!!!\n"); 
getPCICFG = (GetPCICFG)GetProcAddress(hDLL, "PCI_GetPCICFG");

FreeLibrary(hDLL);

View Code

 

 

另外,可以将dll和exe工具放在一起调试,会比较方便,当然,exe里面的用动态调试比较好。静态的可以看前面的链接。

首先,我们写了一个DLL,然后在Solution里面点击添加-->新建项目,这个就是我们新建的exe程序。新建了exe程序后,可以看到Solution里面有了两个项目,这时我们右键exe的项目将其设为启动项目,这时每次调试时都会执行exe的项目,如果要让exe随着DLL的修改而修改,可以同步反应,那么就右键exe项目,为其添加依赖项,选择DLL为他的依赖就好了。这样exe和DLL就可以同步调试了。

 

 

静态调用DLL--同步调试

参考链接:

一、新建一个DLL项目,编译。

二、添加主cpp对应的头文件,然后添加一个“export.h”头文件。

Visual Studio dll java 调用 vs2017调用dll_#include_15

三、修改项目配置

Visual Studio dll java 调用 vs2017调用dll_API_16

四、写主cpp的.h文件

Visual Studio dll java 调用 vs2017调用dll_#include_17

五、写主cpp里面的函数,然后再次编译

Visual Studio dll java 调用 vs2017调用dll_#include_18

六、右键单击左侧列表中“解决方案”,然后在弹出菜单中选择“添加 > 新建项目”,向解决方案中添加一个新的控制台项目,用于测试Bluetooth中导出的printHello()函数是否可以正常访问;

Visual Studio dll java 调用 vs2017调用dll_Code_19

Visual Studio dll java 调用 vs2017调用dll_API_20

七、将控制台项目设为启动项目

Visual Studio dll java 调用 vs2017调用dll_Code_21

八、将exe与dll链接起来,需要对项目进行配置

Visual Studio dll java 调用 vs2017调用dll_Code_22

Visual Studio dll java 调用 vs2017调用dll_Code_23

九、在控制台(exe)属性页窗口中,将配置设置为“所有配置”,然后在左侧“配置属性”列表中,选择“链接器 > 常规”,接着在右侧属性列表中选择“附加库目录”属性右方的编辑框,在弹出的下拉列表中选择“编辑”;$(OutDir)

Visual Studio dll java 调用 vs2017调用dll_#include_24

Visual Studio dll java 调用 vs2017调用dll_#include_25

Visual Studio dll java 调用 vs2017调用dll_Code_26

Visual Studio dll java 调用 vs2017调用dll_#include_27

十、配置完成,在exe中调用dll中的函数。

Visual Studio dll java 调用 vs2017调用dll_Code_28

十一、运行函数Debug\Start Without Debugging

Visual Studio dll java 调用 vs2017调用dll_Code_29

十二、结果

Visual Studio dll java 调用 vs2017调用dll_API_30

大功告成。

 

另外,如果想要dll以c的方式导出,可以修改主cpp的.h文件

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

#pragma once

#ifdef __cplusplus
extern "C" {
#endif


#ifndef Bluetooth_H
#define Bluetooth_H

#include "export.h"

EXPORT_BLUETOOTH void printHello();

#endif //!Bluetooth_H

#ifdef __cplusplus
}
#endif

View Code

 

动态调用DLL--同步调试

 

依次执行静态调用DLL--同步测试的步骤:一、二、三、四、五、六、七、八 

然后在Test_Bluetooth.cpp中调用printHello函数

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

// Test_Bluetooth.cpp : This file contains the 'main' function. Program execution begins and ends there.
//

#include "pch.h"
#include <iostream>
#include <Windows.h>
using namespace std;


class WinDll
{
public:
    WinDll(const char* dll) : mLib(::LoadLibraryA(dll)) {}
    WinDll(const wchar_t* dll) : mLib(::LoadLibrary(dll)) {}
    ~WinDll() { FreeLibrary(mLib); }

    WinDll(const WinDll&) = delete;
    WinDll& operator=(const WinDll&) = delete;

    operator bool() { return !!mLib; }

    template <typename Ret, typename... Args>
    Ret Invoke(const char* name, const Args& ...args)
    {
        auto proc = GetProcAddress(mLib, name);
        typedef Ret(__stdcall* Func)(Args...);
        return (proc) ? reinterpret_cast<Func>(proc)(args...) : (Ret());
    }

private:
    HMODULE mLib;
};


int main()
{
    WinDll bluetooth("Bluetooth.dll路径");
    if (bluetooth) {
        printf("start\n");
        bluetooth.Invoke<void>("printHello");
        bluetooth.~WinDll();
    }
}

View Code

 

如此可行。

 

动态调用封装模板

参考链接:

动态调用dll,用到一个函数就要写那一串,写得很烦,就想有没有简单的方法封装一下,以便调用,之前找到一个封装类WinDll,用一些简单的还不错,用得还蛮开心的,不过当有函数的参数是char*的话就会出现问题,因此又找了一种其他的封装方式。

Visual Studio dll java 调用 vs2017调用dll_API

Visual Studio dll java 调用 vs2017调用dll_Code_02

template<typename _T> class Api;
template<typename _Res, typename... _ArgTypes>
class Api<_Res(_ArgTypes...)> {
public:
    Api(const char* dllName, const char* funcName) {
        _M_module = LoadLibraryA(dllName);
        _M_func = reinterpret_cast<_Func>(GetProcAddress(_M_module, funcName));
    }
    ~Api() {
        if (_M_module) FreeLibrary(_M_module);
    }
    _Res operator()(_ArgTypes... __args) const {
        return _M_func(__args...);
    }
private:
    typedef _Res(*_Func)(_ArgTypes...);
    _Func _M_func;
    HMODULE _M_module;
};

int main(int argc, char* argv[]){
    Api<int(char*, char*)> write_head("F:\\aaa.dll", "print_hello");
    int result = write_head("123", "456");
    write_head.~Api();
}

View Code