《头文件导致Symbol xxx multiply defined重复定义问题分析和解决》

1.问题

main.h

#ifndef _MAIN_H
#define _MAIN_H

unsigned char i;

#endif

main.c

#include "main.h"

main()
{
  ;  
}

a.c

#include "main.h"

略

  然后编译a.c和main.c,就会提示Symbol i multiply defined(by a.o and main.o)

 

2.问题分析

2.1 #ifndef不是已经预防重复编译了?

  #ifndef #define #endif防止的是“重复编译”,而不是“重复定义”。
  重复编译可能造成重复定义,但重复定义的来源不只有重复编译。

  从代码变成可执行的程序,需要两个步骤
  编译和链接
  编译开始时,将所有#include头文件的地方替换成该头文件的代码
  在编译阶段,编译所有源文件成为模块,各模块中的每个变量与函数都得到了属于自己的空间
  在链接阶段,各个模块被组合到一起

  #ifndef能够防止在编译阶段,一段代码被重复编译,并且由此可以避免一个变量被重复定义
  但它不能防止链接阶段,各模块中都有叫某个名字的变量,于是报链接错误:变量重复定义

 

3.解决方法

  不仅用#ifndef组合防止重复编译,而且将变量在源文件中定义,只在头文件里放extern声明。

  这样各模块在编译的时候,就知道“有这么个变量,但它的空间不在我这里”,链接的时候,这个变量虽然出现在所有包含这个头文件的模块里,但只有一个模块是它的真身所在。

 

  

 

 

  

  

posted @ 2020-04-03 10:49  一个不知道干嘛的小萌新  阅读(10267)  评论(0编辑  收藏  举报