转载下,对于VS的编译选项介绍蛮清楚的!!

 

1. 静态链接库、动态链接库、CRT、STL

    我们要到一个函数,要么是需要该函数的源代码,要么是知道该函数的声明并有该函数的实现,这里的“实现”又分为静态链接库、动态链接库。在windows平台上,静态链接库对应以.lib为后缀的库文件,动态链接库对应.dll为后缀的动态链接库文件。关于静态链接库、动态链接库请参考wikipedia相应条目:

http://en.wikipedia.org/wiki/Static_library
http://en.wikipedia.org/wiki/Dynamic-link_library

    我们用VC++写的程序默认编译为可执行文件(.exe),如果想发布自己的库,可以在VS的“项目属性 >> 配置属性 >> 常规 >> 配置类型”修改。这样如果以后想用这些函数就不需要引入对应.cpp文件,而只需包含带有该函数声明的头文件,并引用库文件即可——对于静态链接库,可以#pragma comment (lib, "xxx.lib")”指令,或在VS的“项目属性 >> 配置属性 >> 链接器 >> 输入 >> 附加依赖”中添加;对于动态链接库,可以用“__declspec(dllimport)”声明要用的函数,如果为.dll文件实现了导入库(对应的.lib文件,里面实现了函数导入,使用同静态链接库),则动态库的使用同静态库,只是程序执行时需要.dll文件。msdn上有静态库和动态库的使用教程:

http://msdn.microsoft.com/en-us/library/ms235627.aspx
http://msdn.microsoft.com/en-us/library/ms235636.aspx

    简单总结,可执行文件(.exe)和库文件(.lib、.dll)都含有源代码编译出来的可执行二进制代码。静态链接和动态链接的区别在于:静态链接编译出的可执行代码体积较大,动态链接编译出的可执行代码执行时依赖对应的.dll文件。

    CRT(C语言运行库)实现了C语言相关初始化代码以及实现了C函数库,C++可以看做C语言的超集,所以C++并没有“CPRT(C++运行库)”,C++也使用CRT,标准C++除CRT外还实现了STL(standard C++ library,C++标准库,注意STL是Standard Template Library的缩写,因为C++标准库主要是用模板实现的)。既然函数的“实现”至少有静态和动态之分,那CRT或STL也有不止一个版本,后文针对VC2010平台讨论这些版本。

    总结,CRT是C语言函数库及初始化代码的实现,STL是C++标准库的实现,所谓“实现”就是由源代码编译出来的.lib、.dll文件等。

 

2. VS的编译选项

    在VC2010上,CRT和STL至少分为静态和动态,静态和动态中又各自有Debug和Release版本(早期VC还有单线程和多线程之分,目前VC++中只提供多线程版本),这样CRT和STL都有至少四个版本。现在来解释引言中的符号未定义、符号重定义链接错误的可能情景,程序A中调用了函数f,函数f是在程序B中编写的,为了使用f,将程序B编译为库(而非.exe)——静态库:B.lib\动态库:B.lib、B.dll,程序A为了使用f,包含头文件B.h(其中有函数f的声明)并引用B.lib:

1 #include"B.h"
2 #pragma comment (lib, "B.lib")

    如果没有上面的第二句代码,则出现了符号未定义的链接错误:

main.obj : error LNK2019: 无法解析的外部符号 _f@0,该符号在函数 _main 中被引用

    上面错误信息中的“_f@0”具体取决于函数调用约定的命名方式(_cdecl、_stdcall等)。

    如果编译程序B时使用了动态版本的CRT而编译A时使用的是静态版本CRT(即A、B使用了不同版本的CRT),则出现了符号重定义之类的链接错误(不绝对)。

    当然如果用动态链接版本的B,程序A运行时可执行文件搜索路径中必须包含B.dll,否则报告“丢失xxx.dll”之类的错误。

    设置程序到底使用哪个版本的CRT可在VS的“项目属性 >> 配置属性 >> C/C++ >> 代码生成 >> 运行库”中设置,现在将几种设置对应的库文件,编译器的宏定义列在下表:

Option

Preprocessor directives

C run-time library (without iostream or standard C++ library)

Standard C++ Library

/MT

_MT

libcmt.lib

LIBCPMT.LIB

/MD

_MT, _DLL

msvcrt.lib (import library for MSVCR100.DLL)

MSVCPRT.LIB (import library for MSVCP100.dll)

/MTd

_DEBUG, _MT

libcmtd.lib

LIBCPMTD.LIB

/MDd

_DEBUG, _MT, _DLL

msvcrtd.lib (import library for MSVCR100D.DLL)

MSVCPRTD.LIB (import library for MSVCP100D.DLL)

    其中,MT为是multi-thread的缩写,上面说了,所有这些库都是多线程的,大写D代表DLL,小写d代表debug,如/MDd下引用动态链接调试版本的库,并且编译器定义宏_DEBUG, _MT, _DLL(程序中可以用#ifdef指令来判断库版本),引用的CRT实现文件为MSVCPRTD.LIB,该文件只是导入库并没有具体的执行二进制代码,程序运行时动态链接MSVCP100D.DLL文件,STL实现文件同理。

    文件名“MSVC[R,P]100[D]”中的“100”对应VC2010,VC2003、VC2005、VC2008、VC2010、VC2012分别为71、80、90、100、110,有些时候我们运行一个程序提示“丢失msvcrxxx.dll”,可以通过安装对应VS来解决,如果不想安装VS,也可通过安装“Microsoft Visual C++ 20xx [SP1] Redistributable Package”来解决。

    可参考msdn的C run-time libraries条目:

http://msdn.microsoft.com/en-us/library/vstudio/abx4dbyh(v=vs.100).aspx

 

转载自: http://www.cnblogs.com/liangliangh/p/3521381.html

posted on 2017-08-08 06:15  池的巧克力  阅读(1740)  评论(0编辑  收藏  举报