【环境】

  > KEIL5.25

  > win10

  > @2018-4-23

【问题】

头文件互包含导致的错误(使用了另一文件的类型定义)

文件<fileA.h>

  <fileA.h>

  #ifndef    __FILEA_H__
#define __FILEA_H__ #include "fileB.h" typedef struct
{
int var;
}stuct_A_s; #endif

文件<fileB.h>

 <fileB.h>

 #ifndef    __FILEB_H__
#define __FILEB_H__ #include "fileA.h" typedef struct
{
int var;
struct_A_s obj;
}struct_B_s; #endif

编译后报错:在文件<fileB.h>中   error:  #20: identifier "struct_A_s" is undefined

【分析】

导致上述错误的原因,是在文件<fileA.h>中,使用了 fileB.h,而文件<fileB.h>中使用的类  struct_A_s 在其定义之前,故产生了先使用后定义的语法错误

具体分析:

  从文件<fileA.h>开始分析:

  > 执行避免头文件重复包含的宏 --- <fileA.h>

  > 包含文件<fileB.h>,进入文件<fileB.h>

  > 执行避免头文件重复包含的宏 --- <fileB.h>

  > 包含文件<fileA.h>,进入文件<fileA.h>

  > 由于避免重复包含宏的控制,进不去A文件内容部分,跳转回文件<fileB.h>

  > 执行结构体 struct_B_s 定义,结构体成员类型使用了结构体 struct_A_s ,这就出现了使用了未定义的情况,执行完毕跳转回文件<fileA.h>

  > 执行结构体 struct_A_s 定义,结合上一步就发生了先使用后定义的问题,执行完毕结束

  结论:编译器先从文件<fileA.h>开始编译就会出现本文所示错误

  从文件<fileB.h>开始分析:

  > 执行避免头文件重复包含的宏 --- <fileB.h>

  > 包含文件<fileA.h>,进入文件<fileA.h>

  > 执行避免头文件重复包含的宏 --- <fileA.h>

  > 包含文件<fileB.h>,进入文件<fileB.h>

  > 由于避免重复包含宏的控制,进不去B文件内容部分,跳转回文件<fileA.h>

  > 执行结构体 struct_A_s 定义,执行完毕跳转回文件<fileB.h>

  > 执行结构体 struct_B_s 定义,执行完毕结束

  结论:编译器先从文件<fileB.h>开始编译就不会报错误

【解决】

  # 去除文件<fileA.h>中包含文件<fileB.h>的语句部分,可解决此问题

  # 由于分析推出是编译顺序导致错误的出现,特做了一下一些事做实验:

    > 控制文件 <fileA.h> 与 <fileB.h> 的编译顺序  (做了文件名的更改,即按照字母表顺序修改文件名达到两次编译时两个文件的排序相异)

    > 两次编译的结果都是报相同的错误,error:  #20: identifier "struct_A_s" is undefined

【结论】

  # 综上暂推出,编译器在编译时,每个文件都会单独编译一遍,所以不论文件 <fileA.h> 与 <fileB.h> 的排序,都会出现相同的问题

  # 一般是杜绝文件互相包含的,因为文件包含的意义就是要使用被包含文件的一些定义,互相包含就会出现先使用后定义 的情况发生

  # 了解一些编译原理的知识后,将会得到更权威的解惑,以上只是根据现象分析的结果

05-03 23:58