简要的介绍一下在微软开发工具中(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. 20145207 《Java程序设计》第10周学习总结

    前言:   最后一篇java博客好激动啊..不过猜猜我在干什么?没错,安虚拟机,唉!紧接着又是一大波信安系统的博客,真开心~好啦边敲博客,边装虚拟机. 教材知识汇总 13.1 网络概述 13.1.1计 ...

  2. 封装upload文件上传类

    <?php //封装php中的单文件(图片)上传类 /*  //参数1:$file 文件数组  5个属性值 name,type,size,tmp,error   //参数2:文件保存的路径$pa ...

  3. HDU 4064 Carcassonne(插头DP)(The 36th ACM/ICPC Asia Regional Fuzhou Site —— Online Contest)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4064 Problem Description Carcassonne is a tile-based ...

  4. mysql explain

    我们使用EXPLAIN解析SQL执行计划时,如果有下面几种情况,就需要特别关注下了: 首先看下 type 这列的结果,如果有类型是 ALL 时,表示预计会进行全表扫描(full table scan) ...

  5. unable to connect to the virtual device Genymotion 神器启动问题

    截图: 解决方法:win7以上用户在桌面找到:网络--右键(属性)--更改适配器设置--VirtualBox Host-Only Network--属性--双击:Internet 协议版本4(TCP/ ...

  6. java 虚拟机工具

    jps 命令 可以查询开启了rmi服务的远程虚拟机进程状态. -v jvm参数. jstat -gcutil命令 [cangyue@/System/Library/Frameworks/JavaVM. ...

  7. oracle的表空间

    创建表空间 1:创建单个文件的表空间 CREATE TABLESPACE SAMPLE LOGGING DATAFILE 'D:\11.dbf' SIZE 5M REUSE EXTENT MANAGE ...

  8. SSAS的维度表之间的关系只能有一个不能有多个

    我们在SSAS中创建维度的时候,有时候可能一个维度需要用到多个表的字段作为维度属性,那么这多个表之间势必存在关联关系,但是切记维度表之间的关联关系有且只能有一个不能有多个,下面我们来看一个例子. 现在 ...

  9. NOIP200902分数线划定

    NOIP200902分数线划定 描述 世博会志愿者的选拔工作正在 A 市如火如荼的进行.为了选拔最合适的人才,A 市对所有报名的选手进行了笔试,笔试分数达到面试分数线的选手方可进入面试.面试分数线根据 ...

  10. BI案例:BI在连锁零售业应用(ZT)【转】

    第一部分:连锁零售企业上BI的必要性. 目前国内的连锁零售行业的发展趋势,呈现出产业规模化,经营业态多样化,管理精细化的特点.所谓管理精细化就是"精耕细作搞管理,领先一步订系 统" ...