zoukankan      html  css  js  c++  java
  • 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

    测试结果 
    这里写图片描述

  • 相关阅读:
    iOS AppStore个人开发者账号申请
    一个工程多个Target
    React Native环境搭建(iOS、Mac)
    vuex的简单使用
    在vue中使用sass
    一个javascript继承和使用的例子
    在vue中使用Element-UI
    CSS3 Flex布局和Grid布局
    (...)ES6三点扩展运算符
    h5 video切换到横屏全屏
  • 原文地址:https://www.cnblogs.com/zhaiyf/p/8548963.html
Copyright © 2011-2022 走看看