网上、一些书上也写道,这两种方式作用一样。其实仔细分析,它们两者还是有非常大的差异的。

Additinal Dependencies和#pragma comment(lib,"*.lib")都是针对.lib文件的。

而.lib文件其实有两种。

一种是与DLL相关的,描述DLL中的函数入口地址等的,在这里两种方法作用一样。

一种是静态链接库,就是将静态库工程当前所有目标文件编译成一个二进制文件(通常是以.lib为扩展名)。Additinal Dependencies和#pragma comment(lib,"*.lib")在这静态库这里会体现出重大的不同。

假如我们有4个工程,静态库工程A,B,C和EXE工程D,B依赖A,C依赖B,D依赖C,D依赖于C。

#pragma comment(lib,"*.lib")

其作用是放置1个库的链接搜索记录到目标文件中,这个搜索记录排在默认搜索记录之后。可以在一个目标文件中放置多个搜索记录,其顺序依据于源码中添加的顺序。因为这行代码的作用只是指明链接时去相应的lib文件中查找相应函数的实现代码,只是一个记录指明的作用,并没有链接的作用。

lib工程并没有链接过程,只有编译过程。只有在生成DLL/EXE时,才会有链接过程的产生,将会将所有目标链接在一起。

因为没有链接过程,所以在编译B,C, D工程时并不需要指明其依赖的静态库(lib),只需要#include相应的头文件即可以编译生成lib文件。

那么在链接生成DLL/EXE时,则需要指明链接搜索记录。

可以在工程D中指明所有需要链接搜索的lib文件。

#pragma comment(lib,"A.lib")

#pragma comment(lib,"B.lib")

#pragma comment(lib,"C.lib")

这个时候才会真正完成链接过程,会检查函数是否定义了,是否重复定义等链接错误。

因为D表面看只依赖C,所以指明B.lib, C.lib感觉不太直观,更直观的可能是用到哪个lib添加哪个:

工程B中添加#pragma comment(lib,"A.lib")

工程C中添加#pragma comment(lib,"B.lib")

工程D中添加#pragma comment(lib,"C.lib")

Project Property->Additinal Dependencies->*.lib

这种方式附加lib文件,看起来效果与上面的方式一样,其实工作原理是不一样的,认识不清,容易产生一些问题。

AdditinalDependencies->*.lib分两种情况:

1、静态库工程中,Configuration Properties->Librarian->General->Additinal Dependencies

其作用是附加指定的lib文件至当前工程,并编译生成至当前的lib文件中。这种行为有点类似链接,但不是链接,只是将将指定lib文件合并到新的lib文件中来。

如果单独编译A, B, C, 它们均生成50K大小的lib文件。

工程B中Additinal Dependencies->A.lib

工程C中Additinal Dependencies->B.lib

那么B.lib将会是100K,C.lib将会是150K。可以看到是直接合并进来的。

但如果是如下设置:

工程B中Additinal Dependencies->A.lib

工程C中Additinal Dependencies->A.lib;B.lib

那么B.lib将会是100K,C.lib将会是200K。可以看到C.lib直接将A.lib,B.lib直接合并进来了,相当于合并了两份A.lib。这样会导致重复合并,VS编译器会提出warning LNK4006,already defined in *.obj;second definition ignored。虽然报出来的是一个链接警告,但其实并不是链接过程的。因为这样的情况很多,所以会出现非常多这类警告。MSDN对这类警告的有个相差描述:You can get this warning if you try to merge twoimport libs into one.就是指我们上面合并两个lib文件。

解决方法也分两种:

A、不合并多个lib文件成1个lib文件,则我们不需要在静态库工程中设置Additinal Dependencie。可以在DLL/EXE中通过

#pragma comment(lib,"*.lib")一次全部指明lib搜索记录,也可以在DLL/EXE工程中通过Additinal Dependencie一次附加所有涉及到的lib。如果想更直观点,用到哪个lib指明哪个,即分别在相应的lib工程里用#pragma comment(lib,"*.lib")指明相应的lib文件即可。

B、合并多个lib文件,工程B Additinal Dependencies->A.lib, 工程C Additinal Dependencies->B.lib即可。

2、DLL/EXE工程中,Configuration Properties->Linker->Input->AdditinalDependencies

可以看出,这里是在链接器下面的一个设置,也说明这里的附加是为链接完成的,即指明链接lib搜索记录,作用与#pragma comment(lib,"*.lib")一样,只不过在DLL/EXE中会产生链接过程。静态库中出现的重复重复定义、定义未实现均不会报错,但是链接过程中会检查这类错误,所以很多静态库中的错误会在DLL/EXE工程的链接结果中报出来。

Additinal Dependencies和#pragma comment(lib,"*.lib")的分析的更多相关文章

  1. #pragma comment使用

    编程经常碰到,理解的总不是很透彻,在这里查阅资料总结一下! 在编写程序的时候,我们常用到#pragma指令来设定编译器的状态或者是指示编译器完成一些特定的动作. #pragma once : 这是一个 ...

  2. pragma comment的使用

    该宏放置一个注释到对象文件或者可执行文件. #pragma comment( comment-type [,"commentstring"] ) comment-type是一个预定 ...

  3. pragma comment的使用 pragma预处理指令详解

    pragma comment的使用 pragma预处理指令详解   #pragma comment( comment-type [,"commentstring"] ) 该宏放置一 ...

  4. pragma comment的使用(转)

    #pragma 的使用 尽管 C 和 C++ 都已经有标准,但是几乎每个编译器 (广义,包含连接器等) 扩展一些 C/C++ 关键字. 合理地应用这些关键字,有时候能使我们的工作非常方便.下面随便说说 ...

  5. #pragma comment 的使用方法

    转发:https://blog.csdn.net/liruda/article/details/2230617 #pragma comment ( lib,"wpcap.lib" ...

  6. lib包含# #pragma comment

    #pragma comment(lib,"d2d1.lib")#pragma comment(lib,"dwrite.lib")#pragma comment( ...

  7. 在头文件#pragma comment(lib,"glaux.lib");编译器提示waring C4081: 应输入“newline“

    在头文件#pragma comment(lib,"glaux.lib");编译器提示waring C4081: 应输入“newline“ #行不能加分号的

  8. pragma comment

    pragma指令简介 在编写程序的时候,我们经常要用到#pragma指令来设定编译器的状态或者是指示编译器完成一些特定的动作. 下面介绍了一下该指令的一些常用参数,希望对大家有所帮助! 一. mess ...

  9. #include 和 #pragma comment 的相对路径起点

    #include 是以当前文件所在路径为当前目录 #pragma comment 是以当前工程所在路径为当前目录 #include "../../../../ThirdParty/Inclu ...

随机推荐

  1. 从零开始一个http服务器(六)-多路复用和压力测试

    从零开始一个http服务器(六)-多路复用和压力测试 代码地址 : https://github.com/flamedancer/cserver git checkout step6 运行: make ...

  2. 后台运行spark-submit命令的方法

    在使用spark-submit运行工程jar包时常常会出现一下两个问题: 1.在程序中手打的log(如System.out.println(“***testRdd.count=”+testRdd.co ...

  3. 20155217 《Java程序设计》第三次实验报告

    20155217 <Java程序设计>第三次实验报告 实验内容 XP基础 XP核心实践 相关工具 实验要求 没有Linux基础的同学建议先学习<Linux基础入门(新版)>&l ...

  4. 20155226 2016-2017-2 《Java程序设计》第10周学习总结

    20155226 2016-2017-2 <Java程序设计>第10周学习总结 教材学习内容总结 网络编程 网络编程就是在两个或两个以上的设备(例如计算机)之间传输数据. 程序员所作的事情 ...

  5. 20155330 实验四 Android程序设计

    20155330 实验四 Android程序设计 实验内容 1.基于Android Studio开发简单的Android应用并部署测试; 2.了解Android.组件.布局管理器的使用: 3.掌握An ...

  6. 20155338 2016-2017-2 《JAVA程序设计》课程总结

    ---恢复内容开始--- 20155338 2016-2017-2 <JAVA程序设计>课程总结 每周作业汇总: 预备作业一 预备作业二 预备作业三 <JAVA程序设计>第一周 ...

  7. [FJOI2016]建筑师 斯特林数

    早期作品,不喜轻喷. LG传送门 组合数与斯特林数的基本应用. 组合数 大家应该都熟悉它的表达式,但我们这里使用它的递推式会更加方便,下面推导组合数的递推式.设\(\binom{n}{m}\)表示在\ ...

  8. ABP中module-zero快速集成微信用户认证

     https://personball.com/abp/2019/01/01/introduce-abp-module-zero-external-authenticate 

  9. LJ语录

    "保持安静,不要打扰我睡觉." ( 半分钟后) "哎呦喂~可以睡觉喽~" "考场上遇到这种题目,你们应该高兴." "因为考场上那群 ...

  10. oracle数据库之组函数

    组函数也叫聚合函数,用来对一组值进行运算,并且可以返回单个值 常见的组函数: (1)count(*),count(列名)  统计行数:找到所有不为 null 的数据来统计行数 (2)avg(列名)  ...