其实,并不是C语言标准不允许在头文件中包含变量定义,而是编译器本身产生了重复定义的错误。
这时候有些人会很奇怪,我不是写了#ifndef #define #endif这样的命令了吗?
如果你这样以为,就和笔者一开始犯了同样的错误,没有深入地理解#ifndef,#ifndef只是预编译,还没有开始真正的编译,他只能防止同一个编译单元下,重复include同一个头文件而导致的重复定义。
当一个C或者CPP文件在编译之前,预处理器会首先递归包含头文件,形成一个含有所有必要信息的单个源文件,这个源文件就是一个编译单元,一般来说,一个C或CPP文件就是一个编译单元。
让我们看一下示例
头文件(test.h):
- #ifndef _TEST_H_
- #define _TEST_H_
- int i;
- #endif
文件1(file1.c):
- #include "test.h"
- #include "test.h"
- void main()
- {
- i = 0;
- }
这时如果你编译是不会产生任何问题的,因为这是一个编译单元,编译器在第一次定义了_TEST_H_之后,就不会再去定义,也就不会重复定义变量i,但如果是下面这种情况,编译器就会报错;
头文件(test2.h):
- #ifndef _TEST_H_
- #define _TEST_H_
- int i;
- void add(void);
- #endif
文件1(test1.c):
- #include "test2.h"
- void main()
- {
- i = 0;
- add();
- }
文件2(test2.c):
- #include "test2.c"
- void add()
- {
- i++;
- }
这是编译就会报错,因为test1.c和test2.c会产生两个编译单元,在每个编译单元中都会有一个被定义的i,在链接时,编译器就会产生错误。
所以,为了避免上述情况的发生,我们不在头文件中定义变量。如果你想一个变量被多个C文件使用,那么就在C文件中定义好,并在这个C文件的头文件中,使用extern声明一下。