关于形如--error LNK2005: xxx 已经在 msvcrtd.lib ( MSVCR90D.dll ) 中定义--的问题分析解决

本文讨论了在Windows平台上使用Microsoft Visual C++ (MSVC) 编写C++程序时,关于程序库(.lib)与动态库(.dll)之间的链接问题,特别是与C运行时库(CRT)的冲突。文章详细解释了CRT的不同版本(静态链接与动态链接),并提供了解决库冲突的策略。包括如何通过项目属性设置选择静态链接或动态链接的CRT版本,以及使用第三方工具如Dependency Walker辅助诊断和解决冲突。此外,文章还提出了生成程序执行文件时应考虑的策略,如提供必要的DLL文件或使用静态链接方式生成程序。


关于形如--error LNK2005: xxx 已经在 msvcrtd.lib ( MSVCR90D.dll ) 中定义--的问题分析解决

转自:http://hi.baidu.com/qinfengxiaoyue/item/ff262ccfb53b4c2ba0b50a89

引自:http://blog.youkuaiyun.com/sptoor/archive/2011/02/23/6203376.aspx

对全文有改动。

提示:阅读前需要对静态库(.lib)、动态库(.dll)、导入库(这个还是.lib)之间的区别联系有了解。建议先读读宋宝华先生的《VC++动态链接库(DLL)编程深入浅出 》

关键词:程序库、CRT、LIBCMTD.lib、MSVCRTD.lib、链接库的冲突问题、MSVC链接处理

1.问题引出

很久没有写程序设计入门知识的相关文章了,这篇文章要来谈谈程序库 (Library) 链接,以及关于 MSVC 与 CRT 之间的种种

恩怨情仇。

如果你使用的操作系统是 Linux、Mac 或其他非 Windows 平台,你可以忽略这篇文章;如果你使用的操作系统是

Windows 平台,但没有用 Microsoft Visual Studio C++(以下简称为 MSVC)软件撰写 C++ 程序的话,这篇文章对你的

帮助可能很有限;但如果你的操作系统是 Windows,而且你使用的程序整合开发环境是 MSVC 软件撰写 C++ 程序的话,这

篇文章应该能够帮助你釐清一些重要的基础观念。

身为程序设计者,在学习程序设计的过程中,你是否曾经遇过某些看起来不知所云的错误信息,却不知该如何解决?例如当你

快快乐乐地写完程序,并且确认所有的程序代码都能成功通过编译之后,接着执行「生成解决方案」(Build Solution) 的步

骤,结果却跑出一堆莫名其妙的错误:

1>libcmtd.lib(errmode.obj) : error LNK2005: ___set_app_type 已经在msvcrtd.lib(MSVCR90D.dll) 中定义

1>libcmtd.lib(dbgrptw.obj) : error LNK2005: __CrtDbgReport已经在msvcrtd.lib(MSVCR90D.dll) 中定义

1>msvcrtd.lib(MSVCR90D.dll) : error LNK2005: __setmbcp 已经在libcmtd.lib(mbctype.obj) 中定义

1>LINK:warning LNK4098:默认库“msvcrtd.lib”与其他库的使用冲突;请使用/NODEFAULTLIB:library

1>msvcrtd.lib(cinitexe.obj) : warning LNK4098: 默认库“libcmtd.lib”与其他库的使用冲突;请使

用/NODEFAULTLIB:library

.....................

以一般的情况来说,如果在你的程序项目中有使用某些由他人所撰写的第三方程序库或是开源项目的程序库,比较容易会发生

上述的错误状况。从上述这些看似离奇而令人摸不着头线程的错误信息中,我们大概可以猜测问题点应该在于 LIBCMTD.lib

与 MSVCRTD.lib 这两个程序库身上。但到底什么是 LIBCMTD.lib 和 MSVCRTD.lib?在我们的程序代码中有使用这些程序

库吗?

答是肯定的。

2.C Runtime Library(C运行时库)

熟悉 C 语言的程序设计者都知道,如果要使用 printf()、scanf() 或者 fopen() 等等 C 语言的基本 I/O 操作函数时,首先

必须用 #include 语法将 stdio.h 这个标头文件纳入我们的程序代码中。藉由 stdio.h 中对这些 I/O 操作函数所做出的函数

声明 (function declaration),编译器 (Compiler) 才得以确认 printf、scanf 以及 fopen 等等都是合法可用的函数。

而当我们撰写的程序代码经过编译器产出 OBJ 形式的文件之后,需要再经由链接器 (Linker) 的处理程序,将程序代码中全部

有使用到的函数定义 (function definition) 链接生成起来,才能够产生出最后的程序执行文件。问题来了,我们知道 printf、

scanf 以及 fopen 的函数声明存在于 stdio.h 当中,但是这些傢伙的函数定义,也就是真正的定义程序代码,究竟存放在什么

地方呢?

在 C 语言的标准程序库中。

由 C 语言所制订的标准程序库,称之为「执行阶段程序库」,也就是 C Run-Time Library,通常可简称为 CRT。在 C 语

言的标准程序库中,包含了一组常用的基础函数,例如 I/O 处理与字串操作程序等等,所以只要我们使用 C 语言撰写程序代

码,就一定要将编译完成后的程序代码 OBJ 文件,链接至 C 语言的执行阶段程序库,才能够产生出合法的 C 语言程序执行文

件。

而 CRT 并非只有单一一种版本存在。事实上,除了可以依「Debug」与「Release」用途分成两个版本之外,两者又可分别

衍生分出「静态链接」与「动态链接」两种形式:

静态链接

LIBCMTD.lib(Debug版本)

LIBCMT.lib

动态链接

MSVCRTD.lib(Debug版本)

MSVCRT.lib

虽然这四个 CRT 版本的用途与使用方式各不相同,但却有个共通的特点,就是它们都是满足执行线程安全需求,可在多执行线

程程序代码中安全使用的程序库版本。事实上,在过去 MSVC 6 的版本中,本来还有另外两个 LIBCD.lib(Debug版本)与

LIBC.lib 程序库,是专门给单执行线程程序使用的 CRT 版本,但是这两个选项自 MSVC 2005 开始就从设定选项中被删除掉

了,所以现在大多数程序设计者使用的都是多执行线程的 CRT 版本。

在程序库链接 (library linking) 的行为中,静态链接和动态链接的分别,在于使用静态链接时,会直接将程序库的函数定义嵌

入执行文件之中,而使用动态链接时,程序库的函数定义则存在于另外的独立文件,通常是 DLL 格式的文件中,然后与程序执

行文件一同发布给使用者。因此在文件的尺寸上,使用动态链接的执行文件文件,通常会比使用静态链接的执行文件文件更小

一些。

使用动态链接 CRT 版本的好处,是能够将经常使用到的标准程序库们独立出来,放在 Windows 的系统文件夹中,以减少我

们生成出来的执行文件文件尺寸。但反过来说,使用动态链接 CRT 版本的缺点也在于这些与执行文件相依为命的 DLL 文件

上。举例来说,如果程序以 MSVC 2005 生成出 Debug 态的执行文件,则此执行文件需要有 msvcr80d.dll 存在才能顺利执

行;如果是 Release 态,则相依于 msvcr80.dll。但是如果你把相同的程序代码拿到 MSVC 2008 上生成,产生出来的执行

文件则相依于 msvcr90d.dll 与 msvcr90.dll 两个不同的 DLL 文件。不同版本的 MSVC,都会有各自不同的相依 DLL 文

件。

在 MSVC 的程序项目中,如何指定程序代码要使用静态链接或者动态链接的 CRT 版本?其实很容易,只要在项目属性的

「C/C++」页面中,选择「程序代码产生」(Code Generation) 子页面,其中有个「执行阶段程序库」(Runtime Library)

的项目,也就是项目中用来设定 CRT 链接版本的地方。其中总共有四个选项,正好对应于上述静态链接与动态链接的四个不同

程序库版本。

多执行线程调试 (/MTd):对应 LIBCMTD.lib

多执行线程 (/MT):对应 LIBCMT.lib

多执行线程调试 DLL (/MDd):对应 MSVCRTD.lib

多执行线程 DLL (/MD):对应 MSVCRT.lib

如果你没有做任何设定就开始生成程序的话,MSVC 的预设选项则会使用动态链接的版本。

 

3.Standard C++ Library(标准C++库)

请注意,以上只是单纯 C 语言的程序库而没有包含 C++ 语言在内。如果你的程序系统中,有包含 C++ 语言的程序代码的

话,那又是另外一回事了。但是在项目属性的页面中,为什么找不到相关的设定选项呢?因为 MSVC 悄悄地帮程序设计者代劳

处理掉了。只要在程序代码中使用 #include 语法纳入任何一个 C++ 的标头文件,例如 iostream 或 fstream,MSVC 就

会在链接器的运作阶段中,自动帮我们链接 C++ 的执行阶段程序库。而 C++ 的执行阶段程序库,同样可分为四个版本:

静态链接

LIBCPMTD.lib(Debug版本)

LIBCPMT.lib

动态链接:

MSVCPRTD.lib(Debug版本):执行文件相依于 MSVCP90D.dll

MSVCPRT.lib:执行文件相依于 MSVCP90.dll

至于程序执行文件使用的是静态链接或者动态链接的版本,就仰赖于 C 语言的版本设定选项了。举个例子来说,如果你撰写了

一个 Debug 组态的 C++ 程序,并且保留项目原先预设的生成选项(动态链接),那么最终生成出来的程序执行文件将会相

依于 MSVCR90D.dll 以及 MSVCP90D.dll 两个 DLL 文件。如果将相同的程序以 Release 组态生成完成,则会相依于

MSVCR90.dll 以及 MSVCP90.dll 二者。

 

4.问题分析及解决

刚学习程序设计的入门者,经常会在满心欢喜地完成一件程序作品并且传给其他人使用时,却发现不能在别人的电脑上启动程

序,其实就是陷入了使用者电脑缺少 DLL文件而无法执行程序的窘境。有三种方法可以解决这个令人困扰的问题:

1 . 使用者的电脑,必须先安装「Visual C++开发套件」(MSVC 2008 或 MSVC 2005 )。

2 . 将所需的 DLL文件,例如 MSVCR90D.dll与MSVCP90.dll,直接附在程序的下载包当中。

3 . 以静态链接方式生成程序执行文件。

当你无法确定自己的程序或别人的程序,是否相依于某些特定的 DLL文件时,有一个非常好用的免费工具程序 Dependency

Walker,可以开启 EXE格式的执行文件或者 DLL格式的动态程序库,然后详细地条列出它们所相依的 DLL文件。

了解了几种不同的 CRT版本选项之后,回到最前面的错误信息问题,相信各位现在应该能够很清楚地理解,原来会发生这些奇

怪的错误状况,是因为程序同时链接了 LIBCMTD.lib与MSVCRTD.lib而造成函数定义版本冲突也就是说,程序链接器已经

在其中一个 CRT的版本中找到所需的函数定义,但此时却又跳出另外一位 CRT,也给了一份相同函数的实现版本,所以链接

器无法判断应该忽略谁并且选择谁。

而这个状况的发生原因,就是你的程序与程序所链接的外部程序库,使用了不同的 CRT版本之故。例如,当你的程序使用了

Lua,自然必须链接至 Lua的程序库 lua5.1.lib,但如果lua5.1.lib是以静态链接版本的 CRT生成而成,而你的程序却是以预

设选项,动态链接 CRT 来生成程序执行文件的话,如此一来就会产生上述这些错误信息了。至此,问题的答已昭然若揭,

方法有二种

其一是将Lua重新以动态链接 CRT 的方式生成出一个新的程序库;

其二则是将自己的程序项目改成以静态链接 CRT 方式生成。

换个角度想,当你身为一位程序库的设计开发者,想要将自己写的东西分享给其他人,但又不想要完全开放自己撰写的程序源

码时,至少可以同时提供以下四种版本的程序库,以妥善满足使用者的各种不同需求:

Debug:动态链接Debug版本

Release:动态链接版本

Debug_Static:静态链接Debug版本

Release_Static:静态链接版本

然而,有时候世界并不会运作得如此理想。在某些特殊的状况下,当我们使用他人所写的第三方程序库时,有时可能只拿得到

其中某个特定的版本,例如 Release_Static版本时,就很有可能会遇到程序库冲突的错误情形。此时就需要视项目的实际需求

而定,可以在项目属性中指定「忽略特定程序库」(Ignore Specific Library)这个选项,让程序代码链接器忽略某些程序库,

以此化解动静程序库或新旧程序库之间的恩怨冲突。

小测验:你所撰写的程序,必须链接某个以静态多执行线程 (/MT) CRT 生成而成的程序库。如果你的程序在 Debug组态下以

多执行线程侦错 (/MTd)选项生成,是否会产生冲突?如果你的程序在 Release组态下以多执行线程 (/MT) 选项生成,是否

会产生冲突?是的话,应该如何解决?

上面的方法还是不行!会出现其他问题的。

以下是我摸索出的最新的解决方法:

首先,所有的lib文件,使用/MTd或/MT编译(注:即静态链接模式)。Debug调试模式使用/MTd,Release模式使用/MT。

然后,在自己的程序中也使用/MTd或/MT编译。这样就不会出问题了。

三种编译链接库的方式:

(1)连接Windows库。针对Win32 API编写的应用程序,上面的方法可能带来新问题,可以忽略libcmt.lib库,即可。如果还

有其他问题,再忽略相应的库。

(2)MFC静态链接。上面的方法就是针对这种链接方式的,所以没问题。

(3)MFC动态链接。没有试过,应该和(1)类似。

最后补充:如果还不行,直接加入/force:multiple编译参数吧。这次之所以没有使用它,也是为了严谨起见。

<think>我们正在解决一个MFC项目中调用Haru库时出现的LNK2005链接错误。根据用户描述,错误涉及到libcmtd.libMSVCRTD.lib之间的冲突。这通常是由于项目中混用了不同版本的运行时库(例如,一个库使用调试版,而另一个使用发布版,或者同时混用了静态和动态链接的运行时库)导致的。参考引用中提到了Haru库的最新版进行了完全重写,可能因此改变了其库的构建方式,导致与MFC项目的兼容性问题。典型的解决方案是确保整个项目(包括所有依赖库)使用相同的运行时库设置。在MFC项目中,我们可以通过以下步骤调整:1.检查Haru库的构建方式:确认Haru库是使用哪种运行时库(Debug/Release,静态/动态)构建的。通常情况下,如果Haru库是用动态链接的运行时库(如/MD或/MDd)构建的,那么我们的项目也要使用相同的设置。2.调整项目的运行时库设置:-VisualStudio中,右键项目->属性->C/C++->代码生成->运行时库。-确保项目配置(Debug/Release)与Haru库的配置匹配。例如:-Debug配置:使用/MDd(如果Haru库在Debug下使用动态链接)或者/MTd(如果使用静态链接,但注意混用可能引发问题)。-Release配置:使用/MD或/MT。通常,为了避免冲突,建议都使用动态链接运行时库(/MD或/MDd),因为这样多个库可以共享同一个C运行时库,减少冲突。3.检查Haru库是否提供了不同运行时库的版本。如果没有,可能需要自己使用与项目相同的设置重新编译Haru库。4.如果项目中还有其他库,也要确保它们使用相同的运行时库设置。5.另外,LNK2005错误也可能是因为符号被多次定义。检查是否重复链接了同一个库(例如,同时链接了Haru库的静态库和动态导入库),或者在不同的库中定义了相同的函数。根据以上思路,我们给出具体的解决步骤:###解决步骤1.**确认Haru库的构建配置**:-如果Haru库是下载的预编译版本,查看其文档或提供的版本信息,确认它是用哪种运行时库(/MT,/MTd,/MD,/MDd)编译的。-如果Haru库是开源的,并且提供了源代码,建议自己使用与项目相同的运行时库设置重新编译Haru库。2.**统一项目的运行时库设置**:-打开MFC项目的属性页。-在Debug配置下,将运行时库设置为与Haru库Debug版本相同的设置(通常,如果Haru库是Debug版本,建议设置为`/MDd`)。-在Release配置下,设置为`/MD`。-如果Haru库是静态链接(/MT或/MTd),那么项目也要使用对应的静态链接(但注意,MFC项目通常使用动态链接MFC,所以可能还需要调整MFC库的使用方式)。3.**重新编译Haru库(如果需要)**:-如果Haru库的预编译版本与项目设置不匹配,需要从源代码编译Haru库。-在Haru库的解决方案中,修改项目属性,将运行时库设置成与你的MFC项目一致。-然后编译得到新的Haru库,再将其链接到你的项目中。4.**检查库的重复链接**:-在项目属性->链接器->输入中,检查是否有重复的库(例如,同时链接了多个不同版本的Haru库)或者冲突的库(如同时链接了静态和动态版本的运行时库)。5.**清理并重新生成**:-清理解决方案,然后重新生成。###注意事项-由于Haru库最新版完全重写了代码,并且舍弃了旧版本的API兼容性,因此确保使用的是新版本的API。-在MFC项目中,如果使用了动态链接MFC,则运行时库通常使用动态链接(/MD或/MDd)以避免冲突。通过以上步骤,通常可以解决LNK2005链接错误。如果问题仍然存在,需要检查具体的错误信息,看是否还有其他冲突的库。§§相关问题§§1.如何在VisualStudio中修改运行时库设置?2.如何从源代码编译Haru库并配置使其适用于MFC项目?3.如何解决MFC项目中多个库之间的运行时库冲突?</think> ### 解决MFC项目中调用Haru库的LNK2005链接错误 #### 错误分析 LNK2005错误通常发生在库冲突时,尤其是当项目同时链接了不同版本的C运行时库(CRT)。在您的案例中: ``` LNK2005: _malloc 已在 libcmtd.lib(dbgheap.obj)MSVCRTD.lib定义 ``` 这表示项目同时链接了: 1. **静态调试版CRT**(`libcmtd.lib`) 2. **动态调试版CRT**(`MSVCRTD.lib`) Haru库的最新版完全重写了代码(舍弃了旧API兼容性)并优化了性能[^1],但其编译设置可能与MFC项目的CRT配置冲突。 --- ### 解决方案(逐步操作) #### 1. 统一CRT库版本(关键步骤) 在**项目属性 → C/C++ → 代码生成**中设置: | 配置类型 | 设置项 | 推荐值 | |---------|--------|-------| | **Debug** | 运行时库 | `/MDd`(动态调试DLL) | | **Release** | 运行时库 | `/MD`(动态发布DLL) | ! 同时确保**所有依赖库**(包括Haru)使用相同的设置重新编译 --- #### 2. 检查库链接顺序 在**链接器 → 输入 → 附加依赖项**中: ``` MSVCRTD.lib; # CRT库在前 libhpdfd.lib; # Haru库在后 ``` --- #### 3. 强制符号单一定义 在**链接器 → 命令行**添加: ```bash /FORCE:MULTIPLE /NODEFAULTLIB:libcmt.lib ``` --- #### 4. 清理冲突的库引用 检查是否有以下情况: - 项目手动添加了`libcmtd.lib` - Haru库编译时使用了`/MTd`(静态链接) - MFC项目其他第三方库使用了静态CRT --- #### 5. 重建Haru库(如果上述无效) 从源码编译Haru: ```cmake cmake -DCMAKE_MSVC_RUNTIME_LIBRARY="/MDd" .. ``` 确保编译时添加:`-DHPDF_SHARED_LIB=ON`(生成动态库) --- ### 验证流程 1. 清理解决方案 → 重新生成 2. 检查链接器输出是否只出现`MSVCRTD.lib`或`libcmtd.lib`之一 3. 运行时测试PDF生成功能(验证TrueType字体嵌入是否正常)[^1] > ⚠️ 注意:Haru最新版放弃旧API兼容性[^1],请确认代码使用新版API
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值