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文件,且不会有任何报错。

【更多参考】

posted @ 2019-04-03 14:58  Z--Y  阅读(2835)  评论(0编辑  收藏  举报