error LNK1169 找到一个或多个多重定义的符号的解决方法
问题描述如下:
有 三个源文件,A.h、B.cpp、C.cpp。
A.h是头文件,其中声明了三个变量a1、a2、 a3。
B.cpp是A.h中所声明的类的实现源代码,C.cpp是主程序文件。B.cpp和C.cpp中均包含头文件 A.h。
在编译时,编译能够通过,但链接时出了问题,出现”error LNK1169: 找到一个或多个多重定义的符号“的错误。
经过分析,确定了这是由于两个实现文件中重复包含了头文件而造成的。可解决方法却始终找不到。
要 注意的是,在这里,在头文件中加入#ifndef……#endif这样的预编译命令是没用的,因为这是防止嵌套包含头文件的,而本例中并没有嵌套包含,是 在两个文件中分别包含。
解决方法:
因为这三个变量在两个实现文件中都要用到,所以一定要包含在A.h中。后来在网上找到了解决方法,其实很简单。
就是在A.h中的三个变量声明前加上extern 关键字,然后在B.cpp中不加extern关键字再次声明这三个变量。于是编译链接顺利通过。
其实这是C++中比较基础的问题。
还有种情况是定义了函数,但在另个文件中准备用#include打开,但是结果还是会出现。
直接包含不就在两个cpp文件中都定义了相同的函数/变量吗,链接时会出现重复定义(你自己试试),所以需要使用extren申明一下即可,他们使用的是同一个实体。
例如:
1、你在a.cpp中定义了一个函数
void func()
{
}
希望在b.cpp中调用,调用前就需要进行声明,格式如下:
extren void func(); //extren
后面根的形式和函数定义形式要完全相同
void mian()
{
func();
}
2、1、你在a.cpp中定义了一个变量
int a;
希望在b.cpp中使用,使用前就需要进行声明,格式如下:
extren int a; //extren
后面根的形式和变量定义形式要完全相同
void main()
{
int b = a;
}
注意:
申明全局变量,全局函数一定要在cpp中申明
其他类引用该全局变量就include该cpp的h文件
然后extern一下就好了
否则容易出现该重复定义错误
这个"容易"是如何解释的呢?
例如A.h中如果申明了全局变量
int Global;
在B.h中include "A.h"
extern int Global;
则你include A.h相当于把A.h中全局变量的申明也include进来了,编译器就会认为是重复定义
所以全局变量和函数申明一定要在cpp中。
另外:
全局变量(对象)定义和全局函数定义一定不能出现在h文件中。类定义、枚举定义、结构体定义都可以。
类里面声明的函数方法,如果只声明,且没有被调用过,那么编译和链接均可以通过。但是,如果只声明,且被调用到,那么就会出现编译通过、链接不通过的错误。个人推荐,每个声明的函数方法都应该要实现。
编译的时候就是语法检测和声明检测(出现未声明的标识符会报错),链接的时候就是定义检测(出现重定义和函数调用时候没有该函数的定义会报错)。
无法解析的外部符号,大多数情况都是由于只声明了函数方法,没有函数方法实现造成的。
番外话:
1、如果你的A.h的声明,在C.cpp中实现,那么生成的是C.obj文件。
2、如果你只有声明的h文件,没有实现的cpp文件,那么不会生成obj文件,且不会有任何报错。