简要的介绍一下在微软开发工具中(VC)静态链接库和动态链接库生成过程中出现的.dll  .lib  .def 和 .exp文件类型。windows平台上可执行文件可能是一个.exe文件也可能四个.dll文件。当然也有一些比较特别的exe或者dll文件,不过他们有其他样式的后缀名比如屏保程序(.scr),ActiveX DLL 用的.ocx 还有各种驱动使用的各种扩展名。这里我们不讨论.com 和一些脚本文件比如.bat .cmd等,虽然他们仍然是可执行文件。

库就是包含着一坨数据和代码的东西,这个东西可以被连接程序或者其他可执行文件使用。库中这些可使用的对象(数据或者函数)使用一些标记标出来。比如在.obj目标文件中一些简单的符号。这里讨论到两种类型的链接库,静态链接库和动态链接库。

你可以认为静态链接库是一堆目标文件(.obj)文件的集合,我们只是把他们简单的打包在了一半静态链接库中。静态链接库都有一个.lib的扩展名。静态链接库不是用来执行的但它可以被链接程序(link.exe)在生成可执行文件的时候使用。在默认的情况下,静态链接库中所有的符号标记对于linker来说都是可见的,即可使用的。当然在编译你的时候你需要对应好头文件和静态链接库,一个静态链接库和.def 或者.exp文件没有任何的关系。

这里一个缺点就是,当我们使用静态链接库链接生成程序的时候,它里面的一些对象比如函数代码都会拷贝到对应的程序当中去,这个工作原理和.obj文件的使用时的工作原理是一样的。对于一些在不同应用程序中可重复利用的代码来说这并不是意见好事:当我们链接的时候,每一个应用程序中都会有一个.obj中使用到的对象的拷贝。

动态链接库(DLL,在Unix世界中被称作共享目标即.so文件)可以帮我们节省内存空间。当我们链接到一个dll的时候,不会有代码拷贝到目标可执行文件里面,但是会有一个引用放在可执行文件里面。当可执行文件被加载执行的时候系统会检查它使用到的dll然后加载这些dll。使用dll我们可以很方便的升级我们的客户端程序,而不用再次更新可执行文件。

但是在链接linking的时候我们还必须有一些信息来知道怎样链接到dll,比如头文件中对应dll中的一些函数的签名。链接程序需要更多的信息,比如dll的文件名,那些符号可用等等,这些信息存在于导出库文件中。导出库的后缀名也是一个.lib。当linker生成.dll文件的时候,他会自动的生成一个导出库.lib文件。导出库被用来分发给那些在开发阶段使用到这个dll的研发人员,更精确的说,是在他们使用link的时候。在使用的时候导出库的使用和静态链接库的使用基本没啥区别。就是使用导出库的程序在执行的时

候需要对应导出库的dll。

好吧!问题看似解决了,但是为什么我们会看到到处都会出现一些.def文件啊?

def文件(module definition file模块定义文件)是用来创建dll和对应的导出库的。在一个.def文件中,你可以指定dll将会导出哪些符号给用户使用。linker会根据def文件的说明来生成dll和lib。一般的,dll的用户不会对def文件感兴趣(我是指使用dll的开发者和使用最终产品的用户)。注意他和静态链接库的不同点,默认的情况下,dll内部的符号是不可见的。我们有方法让他们可见--在def文件中使用exports语句。但是我们还有其他的方法,比如zaidll的源代码中使用__declspec(dllexport)或者在linker的选项中

使用 /EXPORT 选项来导出一个函数等等。事实上现在__declspec(dllexport)使用很多,而def文件很少使用了。使用def文件,你可以指示链接程序linker其他的一些信息而不是导出动作,比如堆栈的大小等等。但是这些选项经常在linker的命令行中标明了。事实上,def文件在早期win16的dll编程中使用现在在win32中我们基本上把它给抛弃了,以后也是如此。

稍等,在某些地方我们还看到一些.exp文件?exp文件就是导出文件(export file)。在前面的讨论中,我们讨论了使用linker去创建dll(中间还有它的导出库)现在,我们假设我们生成两个dll(or just executables)。但是他们都需要调用一些对方中函数,问题出现了。当我们生成a.dll的时候我们需要b.lib;但是b.lib在对应的b.dll生成之前没有生成,而b.dll的生成又需要a.lib。正因如此,微软的解决办法是使用exp文件,或者叫导出文件。在生成两个dll之前,你使用lib.exe(library mangager tool库管理工具)来创建一个.lib和.exp,即,DLL A 的a.lib 和a.exp,现在linker使用a.lib和DLL B 自己的东西去生成b.dll和b.lib。当你回来链接DLL A的时候你就有了b.lib。这里linker需要知道a.dll中需要导出处啥。这些信息都被缓存到了a.exp文件中。linker不需要def文件或者/EXPORT选项,它仅仅是加载a.exp中的信息。a.exp就像a.dll的两个生成过程(lib.exe and linker)的联系者一样。相似的,linker不会再次生成a.lib。总的来说,这种循环调用的情况不会和在我们的程序中出现,因此,希望你不会再你的程序中用到exp文件。

C++导出文件后缀dll,lib,exp,def的更多相关文章

  1. VS2013 c++ 生成和调用DLL动态链接库(.def 方法已验证OK)

    转载:https://blog.csdn.net/zhunianguo/article/details/52294339 .def 方法 创建动态库方法: 创建动态库是生成 .dll .lib 两个个 ...

  2. 45.QT-连接外部dll,lib库导入问题

    dll库问题 查看MZ_Card.dll对应的文档手册,如下图所示: 所以代码写为: typedef BOOL (*Fun)(BOOL IsOpenComm,unsigned long Port, u ...

  3. dll,lib文件的导入

    这里介绍了两种方式调用,不过我一般用的是第一种,比较方便. 1动态库函数的调用,可以采用静态链接的方式 ,主要步骤如下: 1) 包含DLL中导出的头文件. 2) 采用#pragma comment(l ...

  4. windows编译动态链接库,dll+lib的形式

    之前一直在linux上做开发,没怎么关注过windows上如何编译动态链接库.不过一直存疑,为什么windows上的动态链接库是.dll配合.lib使用的,这个又是怎么生成的呢,通过一段时间的查资料和 ...

  5. vs dll lib 使用记录

    今天把学习opengl的项目从一个电脑copy到另一个电脑时候,发生了glu.dll找不到,导致项目起不来的问题.后来网上查找发现, 虽然我使用了静态连接 mt/mtd 编译, 但是有可能lib中并没 ...

  6. VC静态调用DLL(lib)

    1. #pragma comment(lib, "libxml2.lib")#pragma comment(lib, "iconv.lib")#pragma c ...

  7. CMake INSTALL 命令设置exe dll lib的安装位置

    install(TARGETS ${OUT_NAME} RUNTIME DESTINATION ${CMAKE_BINARY_DIR}/bin LIBRARY DESTINATION ${CMAKE_ ...

  8. 动态库dll使用module.def文件导出函数(像静态库一样使用)

    1.新建文件module.def. 2.动态库工程上右键->属性->链接器->输入->模块定义文件编辑它写入module.def 3.下面为module.def实例(smart ...

  9. dll的导入库(lib)输出路径

    创建动态链接库项目A, 它会生成:dll, lib, exp 等,其中 lib & exp 生成的路径设置在:[Linker]->[All Options]->[Import Li ...

随机推荐

  1. CentOS 7 自动更新时间日期

    # ntpdate time.nist.gov 或 # rdate -s time.nist.gov 下列服务器可用 time-nw.nist.gov

  2. Spark on Yarn

    Spark on Yarn 1. Spark on Yarn模式优点 与其他计算框架共享集群资源(eg.Spark框架与MapReduce框架同时运行,如果不用Yarn进行资源分配,MapReduce ...

  3. 机器学习中的范数规则化之(一)L0、L1与L2范数(转)

    http://blog.csdn.net/zouxy09/article/details/24971995 机器学习中的范数规则化之(一)L0.L1与L2范数 zouxy09@qq.com http: ...

  4. zw版【转发·台湾nvp系列Delphi例程】HALCON CheckDifference

    zw版[转发·台湾nvp系列Delphi例程]HALCON CheckDifference unit Unit1;interfaceuses Windows, Messages, SysUtils, ...

  5. zw版【转发·台湾nvp系列Delphi例程】HALCON BinThreshold

    zw版[转发·台湾nvp系列Delphi例程]HALCON BinThreshold unit Unit1;interfaceuses Windows, Messages, SysUtils, Var ...

  6. 【cruch bang】中切换成左手鼠标

    在“右键”菜单->settings->Edit autostart启动的geany编辑器中,最后加内容: xmodmap -e 'pointer = 3 2 1'

  7. js调用后台方法(如果你能容忍执行的后台方法变成一个常量)

    最近一直在做一个电话拨号的系统,系统不大,但是做的时间有点长了.其中用到了一个技术:js调用后台方法.解决这个问题花了不少时间,现如今仍然还有些不明白的地方,今天跟大家分享一下.真正明白的同学欢迎指正 ...

  8. android 应用架构随笔二(定义BaseApplication并配置Application)

    定义BaseApplication并配置Application import android.app.Application; import android.os.Handler; /** * * = ...

  9. 十步完全理解 SQL(转载)

    英文出处:Lukas Eder. 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程序语言.甚至是函数语言(尽管有些人认为 ...

  10. vim中设置自动匹配括号和引号

    vim ~/.vimrc 在.vimrc中添加一下几行 inoremap ( () <LEFT> inoremap { {} <LEFT> inoremap [ [] < ...