Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 199|回复: 0

[默认分类] DLL中导出函数的两种方式(dllexport与.def文件)

  [复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-3-29 09:50:29 | 显示全部楼层 |阅读模式
    DLL中导出函数的声明有两种方式:
    一种方式是:在函数声明中加上__declspec(dllexport);
    另外一种方式是:采用模块定义(.def)文件声明,(.def)文件为链接器提供了有关被链接程序的导出、属性及其他方面的信息。
    方式一:在函数声明中加上__declspec(dllexport)
    /// 在动态链接库程序中
    /// 声明动态链接库(**.dll)的对外接口函数TestFuction
    extern "C" __declspec(dllexport) int TestFuction(int nType,char *strPath,std::vector<string> &vecData)
    {
       ////do anything here////
       return 0;
    }

    /// 在外部希望调用动态链接库的程序中
    /// 加载动态链接库(**.dll)并调用其对外接口TestFuction
    void func()
    {
      //typedef与函数TestFuction类型相同的函数指针为TESTDLL
      typedef int (_cdecl * TESTDLL)(int nType,char *strPath,std::vector<string> &vecData);
      HINSTANCE hmod;
      //加载动态链接库**.dll
      hmod =::LoadLibrary(_TEXT("dll相对路径\\**.dll"));
      if(NULL == hmod)
      {
         TRACE("加载**.dll失败");
      }
      //定义一个与函数TestFuction类型相同的函数指针lpproc
      TESTDLL lpproc;
      //搜索**.dll中函数名为TestFuction的对外接口
      lpproc = (TESTDLL)GetProcAddress (hmod,"TestFuction");
      //如果搜索成功
      if(NULL != lpproc)
      {
         int nType = 0;
         char* strPath = "Data";
         std::vector<string> vecData;
         //通过函数指针lpproc调用**.dll的接口函数TestFuction
         int nResult = (*lpproc)(nType,strPath,vecData);
      }
      //...
      //在恰当的时候释放动态链接库**.dll
      FreeLibrary(hmod);
    }

    方式二:采用模块定义(.def)文件声明
    首先创建 一个DLL程序(DllTestDef)
    在*.cpp中
    int __stdcall Add(int numa, int numb)
    {
         return (numa + numb);
    }
    int __stdcall Sub(int numa, int numb)
    {
         return (numa - numb);
    }
    然后创建一个.def的文件,在里面加上
    ;DllTestDef.lib : 导出DLL函数
    ;作者:----
    LIBRARY DllTestDef
    EXPORTS
    Add @ 1
    Sub @ 2
    最后创建一个测试程序:.cpp文件如下:
    #include <iostream>
    #include <windows.h>
    using namespace std;
    typedef int (__stdcall *FUN)(int, int);
    HINSTANCE hInstance;
    FUN   fun;
    int main()
    {
           hInstance = LoadLibrary("DLLTestDef.dll");
           if(!hInstance)
               cout << "Not Find this Dll" << endl;
           fun = (FUN)GetProcAddress(hInstance, MAKEINTRESOURCE(1));
           if (!fun)
           {
                  cout << "not find this fun" << endl;
           }
           cout << fun(1, 2) << endl;
           FreeLibrary(hInstance);
           return 0;
    }
    说明:
    .def文件的规则为:
    (1)LIBRARY语句说明.def文件相应的DLL;
    (2)EXPORTS语句后列出要导出函数的名称。可以在.def文件中的导出函数名后加@n,表示要导出函数的序号为n(在进行函数调用时,这个序号将发挥其作用);
    (3).def 文件中的注释由每个注释行开始处的分号 (;) 指定,且注释不能与语句共享一行。
    (4)使用__declspec(dllexport)和使用.def文件是有区别的。
    如果你的DLL是提供给VC用户使用的,你只需要把编译DLL时产生的.lib提供给用户,
    它可以很轻松地调用你的DLL。但是如果你的DLL是供VB、PB、Delphi用户使用的,那么会产生一个小麻烦。
    因为VC++编译器对于__declspec(dllexport)声明的函数会进行名称转换,如下面的函数:
    __declspec(dllexport) int __stdcall Add()
    会转换为Add@0,这样你在VB中必须这样声明:
    Declare Function Add Lib "DLLTestDef.dll" Alias "Add@0" () As Long
    @后面的数由于参数类型不同而可能不同。这显然不太方便。所以如果要想避免这种转换,就要使用.def文件方式导出函数了。
      
    http://www.cnblogs.com/enterBeijingThreetimes/arcHive/2010/08/04/1792099.HTML
      
    1. 1.VS2012创建.def文件
    2. 右键工程名->add->New Item->Visual C++->Code->Module-Define File(.def)
    3. 2.def文件的编写
    4. 一个def文件必须有两个部分:LIBRARY和EXPORTS
    5. 让我们先看一个基本的.def文件稍后我将解析
    6. LIBRARY dll_tutorial
    7. DESCRIPTION "our simple DLL"
    8. EXPORTS
    9. Add @1
    10. Function @2
    11. 第一行,""LIBRARY""是一个必需的部分。它告诉链接器(linker)如何命名你的DLL。下面被标识为""DESCRIPTION""的部分并不是必需的,但是我喜欢把它放进去。该语句将字符串写入 .rdata 节[据 MSDN],它告诉人们谁可能使用这个DLL,这个DLL做什么或它为了什么(存在)。再下面的部分标识为""EXPORTS""是另一个必需的部分;这个部分使得该函数可以被其它应用程序访问到并且它创建一个导入库。当你生成这个项目时,不仅是一个.dll文件被创建,而且一个文件扩展名为.lib的导出库也被创建了。除了前面的部分以外,这里还有其它四个部分标识为:NAME, STACKSIZE, SECTIONS, 和 VERSION。我将不再在本文中涉及这些内容,但是如果你在Internet上搜索,我想你将找到一些东西(译注: MSDN2003上对模板定义文件各部分内容有详尽解释,请参阅)。另外,一个分号(;)开始一个注解
    12. 3.def文件路径的设置
    13. Lingker->Input->Module Definition File
    14. 4.def文件的作用
    15. 通俗解释: 在VC++中,生成DLL可以不使用.def文件。只需要在VC++的函数定义前要加 __declspec(dllexport)修饰就可以了。但是使用__declspec(dllexport)和使用.def文件是有区别的。如果 DLL是提供给VC++用户使用的,你只需要把编译DLL时产生的.lib提供给用户,它可以很轻松地调用你的DLL。但是如果你的DLL是供其他程序如 VB、delphi,以及.NET 用户使用的,那么会产生一个小麻烦。因为VC++对于 __declspec(dllexport)声明的函数会进行名称转换,如下面的函数: __declspec(dllexport) int __stdcall IsWinNT() 会转换为IsWinNT@0,这样你在VB中必须这样声明: Declare Function IsWinNT Lib "my.dll" Alias "IsWinNT@0" () As Long @的后面的数由于参数类型不同而可能不同。这显然不太方便。所以如果要想避免这种转换,就要使用.def文件方式。 EXPORTS后面的数可以不给,系统会自动分配一个数。对于VB、PB、 Delphi用户,通常使用按名称进行调用的方式,这个数关系不大,但是对于使用.lib链接的VC程序来说,不是按名称进行调用,而是按照这个数进行调用的,所以最好给出。 例子:我们用VC6.0制作一个dll,不使用.def文件,在头文件中这样写 #ifndef LIB_H #define LIB_H extern "C" int _declspec(dllexport)add(int x,int y); #endif
    16. [url=http://blog.sina.com.cn/s/blog_78fd98af0101ghno.html]http://blog.sina.com.cn/s/blog_78fd98af0101ghno.html[/url]
    17. http://www.cnblogs.com/leijiangtao/p/4805940.html
    复制代码
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-3-29 12:51 , Processed in 0.339745 second(s), 37 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表