error LNK2019 无法解析的外部符号

本文探讨了内联函数的定义位置及其引发的LNK2019错误。通常情况下,内联函数应直接定义在头文件中以避免链接时出现未解析的外部符号错误。

转载地址http://www.programlife.net/error-lnk2019.html

一个常识性错误,关于内联函数的(error LNK2019 无法解析的外部符号)

通常编写函数时,会把声明放到头文件(*.h)中,而把函数定义放在源文件(*.cpp)中,在其他源文件中使用时包含对应的头文件即可。但是内联函数(inline)不要定义在cpp文件中,因为我们通常都是包含头文件,而编译器在编译时会对内联函数进行展开操作,但是编译器只能在包含的头文件里搜索函数的定义体,如果头文件只有声明,显然编译器不知道该如何展开,于是就出错了。

每一个使用内联函数的地方,都应该知道内联函数的定义,所以内联函数就直接定义在头文件中

一个错误的例子:

// Test.h
#ifndef _TEST_H
#define _TEST_H

class CTest
{
public:
    /*inline */void FooBar();
    //{
    //    printf("FooBar() called.\n");
    //}
};

#endif

// Test.cpp
#include "Test.h"
#include <stdio.h>

inline void CTest::FooBar()
{
    printf("FooBar() called.\n");
}

// Main.cpp
#include <stdio.h>
#include "Test.h"

int main(int argc, char **argv)
{
    CTest test;
    test.FooBar();

    return 0;
}
Visual Studio会抱怨:Main.obj : error LNK2019: 无法解析的外部符号 “public: void __thiscall CTest::FooBar(void)” (?FooBar@CTest@@QAEXXZ),该符号在函数 _main 中被引用不过,既然内联函数在编译器就要展开,那为何在链接时才报错呢?这一点不是很明白,微软MSDN是这么说的:
同样,如果项目使用函数内联,但在 .cpp 文件(而非头文件)中定义函数,也会得到 LNK2019。头文件包含在任何被认为合适的位置,但只有在 .cpp 文件通过编译器时函数才内联;因此当函数用于其他模块时,链接器将函数看成无法解析的外部对象。
### 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` 添加到项目中,并正确配置头文件路径。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值