VC C运行时库(CRTL)的几个版本及选用

最近做项目碰到了一个关于在动态库中使用MFC以及在静态库中使用MFC的问题,我的工程在DEBUG编译链接时都没有问题,可是到了RELEASE编译时在链接的时候就出现如下的错误:

nafxcw.lib(array_s.obj) : error LNK2005: "public: __thiscall CStringArray::CStringArray(void)" (??0CStringArray@@QAE@XZ) already defined in mfc80.lib(MFC80.DLL)

查了下原因发现我当前工程是在静态库中使用MFC,而当前工程又依赖其他的一些库,那些库中基本都是选择在动态库中使用MFC,原因应该就在这,将当前设置改为在动态库中使用MFC问题便解决了。究其原因还是对VC的运行时库弄不太明白,之所以出现链接问题,是因为当前工程和它所依赖的库以不同的方式使用MFC,一个是静态链接,一个是动态链接,而且动态链接使用MFC(在RELASE编译时)使用动态运行时库(MSVCRT.lib),而静态链接使用MFC则使用静态运行时库(LIBC.lib(单线程)/LIBCMT.lib(多线程))。

下面是我搜到的关于VC运行时库版本的一些资料,运行时库通常是由编译器厂商提供,是平台相关的。

 

VC C运行时库(CRTL)的几个版本及选用
Michael 2006年7月27日于突尼斯

VC++ C运行时库(以下简称CRTL)是指LIBC.LIB/LIBCMT.LIB/MSVCRT.LIB以及他们对应的DEBUG版本(在名称后面加"D")。
VC++ 4.2以前的版本中CRTL包含了C++的iostream库函数,但是在4.2及以后的版本中(添加了对C++标准库的支持),iostream库函数被独立出来,为支持老的iostream和新的标准iostream函数,4.2及后续版本存在两套iostream库,分别是(老的)LIBCI.LIB/LIBCIMT.LIB/MSVCIRT.LIB和(新的)LIBCP.LIB/LIBCPMT.LIB /MSVCPRT.LIB。针对DEBUG版本,分别存在名称后加"D"的对应库。并且新老的iostream库是不兼容的,不能在同一个应用中混合使用。

一 版本特性列表
对CRTL的几个版本及特性列表(RELEASE&DEBUG Version)如下:
RELEASE版本:

CRTL (without iostream)特性VC编译选项预编译宏
LIBC.LIBSingle threaded, static link/ML
LIBCMT.LIBMultithreaded, static link/MT_MT
MSVCRT.LIBMultithreaded, dynamic link (import library for MSVCRT.DLL),对于不同的VC版本对应的DLL名称不一:
VC1.0-MSVCRT10.DLL
VC2.0-MSVCRT20.DLL
VC4.0-MSVCRT40.DLL
VC4.2-MSVCRT.DLL
VC5.0-MSVCRT.DLL
VC6.0-MSVCRT.DLL
/MD_MT, _DLL

Standard C++ LibraryCharacteristicsOptionDefined
LIBCP.LIBSingle threaded, static link/ML
LIBCPMT.LIBMultithreaded, static link/MT_MT
MSVCPRT.LIBMultithreaded, dynamic link (import library for MSVCRT.DLL),对于不同的VC版本对应的DLL名称不一:
VC4.2-MSVCPRT.DLL
VC5.0-MSVCP50.DLL
VC6.0-MSVCP60.DLL
/MD_MT, _DLL

Old Iostream LibraryCharacteristicsOptionDefined
LIBCI.LIBSingle threaded, static link/ML
LIBCIMT.LIBMultithreaded, static link/MT_MT
MSVCIRT.LIBMultithreaded, dynamic link (import library for MSVCIRT.DLL)/MD_MT, _DLL

DEBUG版本:
CRTL(without iostream)CharacteristicsOptionDefined
LIBCD.LIBSingle-threaded, static link/MLd_DEBUG
LIBCMTD.LIBMultithreaded, static link/MTd_DEBUG, _MT
MSVCRTD.LIBMultithreaded, dynamic link
(import library for MSVCRxD.DLL)1
/MDd_DEBUG, _MT, _DLL
1   In place of the “x” in the DLL name, substitute the major version numeral of Visual C++ that you are using. For example, if you are using Visual C++ version 4, then the library name would be MSVCR40D.DLL.

Standard C++ Debug LibraryCharacteristicsOptionDefined
LIBCPD.LIBSingle-threaded, static link/MLd_DEBUG
LIBCPMTD.LIBMultithreaded, static link/MTd_DEBUG, _MT
MSVCPRTD.LIBMultithreaded, dynamic link (import library for MSVCRTD.DLL),对于不同的VC版本对应的DLL名称不一:
VC4.2-MSVCPRTD.DLL
VC5.0-MSVCP50D.DLL
VC6.0-MSVCP60D.DLL
/MDd_DEBUG, _MT, _DLL

iostream Debug LibraryCharacteristicsOptionDefined
LIBCID.LIBSingle threaded, static link/MLd_DEBUG
LIBCIMTD.LIBMultithreaded, static link/MTd_DEBUG, _MT
MSVCIRTD.LIBMultithreaded, dynamic link (import library for MSVCIRTD.DLL)/MDd_DEBUG, _MT, _DLL


二 单线程(Single threaded)和多线程(Multithreaded)的区别

简单地说,单线程版本提供的CRTL函数不是可重入(Re-entrant)的(只有少部分函数是可重入),多线程版本提供的CRTL函数是可重入的。
对于多线程应用程序来说,如果使用单线程的CRTL将可能导致数据崩溃,因为在同一时间可能有多个线程同时访问CRTL函数中的某个静态数据,这个数据在单线程CRTL中不受保护(如果访问的是栈数据,则没有问题,因为栈的数据在每个独立线程中分配)。所以,此时需要使用多线程的CRTL,如果坚持使用单线程CRTL,应用程序必须亲自对数据共享访问进行保护处理,比如设立临界区。

三 静态链接(Static link)和动态链接(Dynamic link)的区别


采用静态链接的应用程序发布后不依赖于CRTL,同时该库中的代码和数据在该应用程序调用的其他动态库中是访问不到的。
采用动态链接的应用程序必须依赖于CRTL(比如MSVCRT.DLL)。
对于动态链接CRTL的应用程序,在调用库函数时需要遵守两个原则:1)调用习惯为C(__cdecl)习惯;2)函数参数类型为值或者指针类型。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值