“error LNK2019: 无法解析的外部符号”之分析

本文详细介绍了在使用VS2008开发过程中遇到的LNK2019错误的各种原因及解决办法,包括缺少库文件、未正确加入工程、C与C++混编等问题。

转自备用: http://www.cnblogs.com/hiloves/p/4678848.html

 

最近在用VS 2008开发,初学遇到不少问题,最头疼的问题之一就是:LNK2019。

百度一下讲的并不够全面,反正都没解决我的问题。

error LNK2019问题在VC 6.0中是error LNK2001: unresolved external symbol问题,可能错误号改了。

编译时出现类似这样的错误:Dlgcode.obj : error LNK2019: 无法解析的外部符号 _readRegmark,该符号在函数 _AboutDlgProc@16 中被引用。这种错误的本质是链接器无法在已编译的obj、lib或dll文件中找到函数定义。

1、这是百度找到的方法:http://jingyan.baidu.com/article/4d58d54135d7a79dd4e9c0ad.html。就是有头文件(有了函数声明)却没有lib。一般出现于你使用了第三方提供的库,下载了头文件却忘了载库文件,或库文件忘记放到相应的目录下了。

2、你自己写的函数声明的头文件也写了函数定义的cpp文件,却依然出现LNK2019错误。可能原因:忘记将这两个文件加入工程了。一般出现于用Visual Studio和记事本(或UltraEdit)混合开发过程,你用记事本include了相应的头文件,却忘了在Visual Studio的工程中加入它们了。也可能出现于在解决方案的开发过程,在解决方案下的某个工程中加入了它们却忘了在其他工程中加入,我只接触过VC 6和VS 2008,中间好多年没用过新版本VS,到2008时突然发现怎么多了个“解决方案”,“解决方案”下面还可以放好多工程,于是经常在一个工程中写了共享的源代码,却忘了在别的工程中加入它们。这个问题类似于第1个,不同的是这个库是你自己提供的,但没有把它交给VS 2008编译出来。

3、你自己写的函数声明的头文件也写了函数定义的cpp文件也加入工程了而且你很确定函数体肯定是在这个库文件中,却依然出现LNK2019错误。可能原因:C语言和C++语言混编,因为C++支持函数重载所以C++编译器生成的库文件中的函数名会面目全非,例如C编译器会生成 _readRegmark 这个函数名,而C++编译器则生成了"void __cdecl readRegmark(char *)" (?readRegmark@@YAXPAD@Z)这么个函数名。当你的函数是用C语言写的,VS编译器会按C语言规则编译,但链接器却不知道还傻傻的用C++规则的函数名去找结果就找不到了,而你还百般肯定TM的不就在这个库中吗你个睁眼瞎。解决:在C语言的头文件中加入

复制代码
#ifdef __cplusplus
extern "C" {
#endif

void readRegmark(char *regmark);  //这里写函数声明

#ifdef __cplusplus
}
#endif
复制代码

给链接器提示这个函数是C语言的,别TM找错了。

4、这是我百度来的,姑且也放进来。http://blog.youkuaiyun.com/jtop0/article/details/5779782。模板声明和实现要放在同一文件夹中。

5、也是百度来的。http://www.programlife.net/error-lnk2019.html。内联函数定义在头文件中。

6、百度的。http://jingyan.baidu.com/article/d621e8da0d7c022864913f40.html。错误的工程类型造成的。

7、貌似还有不尽之处。http://www.douban.com/note/65638800/

 

### LNK2019 错误的解决方案 LNK2019 是 Microsoft Visual Studio 中常见的链接器错误,表示链接器在尝试创建最终的可执行文件或库时,无法找到某个函数、变量或对象的定义。以下是导致该错误的常见原因及解决方法: #### 1. **未正确实现声明的函数** 如果在头文件中声明了一个函数,但未在源文件中提供其实现,则会导致此错误。确保所有声明的函数都有对应的实现[^4]。 ```cpp // MyHeader.h #ifndef MYHEADER_H #define MYHEADER_H void myFunction(); // 声明 #endif // MySrc.cpp #include "MyHeader.h" void myFunction() { // 实现 // 函数体 } ``` #### 2. **忘记编译或链接某些源文件** 如果项目中包含多个 `.cpp` 文件,但未将所有必要的源文件添加到项目中,则可能导致链接器找不到某些符号的定义。检查项目设置,确保所有相关 `.cpp` 文件都被正确添加[^3]。 #### 3. **静态变量未初始化** 对于全局或静态变量,如果仅在头文件中声明而未在源文件中定义,则会导致此错误。确保为静态变量提供定义[^1]。 ```cpp // MyHeader.h extern int globalVar; // 声明 // MySrc.cpp int globalVar = 0; // 定义 ``` #### 4. **链接库缺失或配置错误** 如果项目依赖于外部库(如 Qt 或 SQLite),但未正确配置链接器选项或未指定库路径,则会引发此错误。确保在项目属性中正确配置了库路径和依赖项[^5]。 - 在 Visual Studio 中: - 菜单栏选择 `项目 -> 属性 -> 配置属性 -> 链接器 -> 输入` - 在 `附加依赖项` 中添加所需的库文件名。 - 在 `常规 -> 附加库目录` 中添加库路径。 #### 5. **名称修饰问题** C++ 编译器会对函数名进行修饰(name mangling),以支持重载等功能。如果使用 C++ 编写代码但调用了用 C 编写的库,则需要通过 `extern "C"` 关键字禁用名称修饰[^4]。 ```cpp extern "C" { void cFunction(); // 禁用名称修饰 } ``` #### 6. **入口点问题** 在某些情况下,错误信息可能显示为 `error LNK2019: 无法解析外部符号 WinMain`。这通常是因为项目类型被错误地设置为 Windows 应用程序,但代码中未定义 `WinMain` 函数。检查项目属性中的 `子系统` 设置[^2]。 - 如果项目是控制台应用程序: - 将 `子系统` 设置为 `Console (/SUBSYSTEM:CONSOLE)`。 - 如果项目是图形界面应用程序: - 确保定义了 `WinMain` 函数。 ```cpp int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { // 入口点代码 return 0; } ``` #### 7. **重复定义或冲突** 有时,多个文件中可能存在同名的全局变量或函数定义,导致链接器混淆。检查是否有重复定义,并确保每个符号在项目中唯一[^3]。 --- ### 示例代码 以下是一个简单的示例,展示如何避免 LNK2019 错误: ```cpp // MyHeader.h #ifndef MYHEADER_H #define MYHEADER_H class MyClass { public: MyClass(); void displayMessage(); }; #endif // MySrc.cpp #include "MyHeader.h" #include <iostream> MyClass::MyClass() {} void MyClass::displayMessage() { std::cout << "Hello, World!" << std::endl; } // Main.cpp #include "MyHeader.h" int main() { MyClass obj; obj.displayMessage(); return 0; } ``` 确保将 `MySrc.cpp` 添加到项目中,并正确配置头文件路径。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值