现在的位置: 首页 > 综合 > 正文

#ifndef#define#endif

2013年10月26日 ⁄ 综合 ⁄ 共 2002字 ⁄ 字号 评论关闭

参考:http://blog.csdn.net/q191201771/article/details/6399820

#ifndef <标识> 

<标识>在理论上来说可以是自由命名的,但每个头文件的这个“标识”都应该是唯一的。标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线,如:stdio.h

#ifndef _STDIO_H_
#define _STDIO_H_

......

#endif

这里首先说明下几点基础知识, 相信大部分人对于以下几点大部分都已经知道了, 你也可以直接跳到最后部分看#ifndef#define#endif的真正作用

  1.预编译阶段把所有#include ”***.h“ (“”与<>的区别这里就不说了)用***.h的内容来替换了, 所以之后就没有.h了所有.h的内容都已经包含进了需要它们的.cpp中(注:该步个人认为是发生在预编译阶段) 

  2.生成最后的exe文件是由编译、链接两步完成的, 编译是源代码生成obj二进制目标文件的过程, 注意一个源代码文件(指.cpp, 而非.h, .h已经被包含进.cpp中了)生成一个obj文件, 在VS2005中单独编译一个obj的方法是选中该.cpp文件ctrl+f7, 文章中以下所说的编译皆按该方法执行而非F7, 由于编译是独立的, 所以在两个独立的编译单元里是可以有重名的函数的, 例如a.cpp中可以有一个void fun(); b.cpp中可同时有一个void
fun(); 这点十分重要, 大家可以试一下并且理解清楚

  3.编译期间, 我们只要声明了的东西就能使用, 而无需它的定义, 声明可以重复, extern在编译时是告诉该编译单元该变量的定义在别的编译单元里, 相当于声明, 链接时, 定义在整个程序中有且仅有一份,  例如如下代码, 编译可通过,但链接时失败

  1. extern int a;  
  2. //extern double a;//错误, a只能有一个类型  
  3. extern int a;  
  4. extern void fun1(int a, int b);  
  5. extern void fun1(int a, int b);  
  6. extern void fun1(int a, int b, int c);//函数重载  
  7. void fun2();  
  8. void fun2(int a);//函数重载  
  9. void fun2()  
  10. {  
  11. }  
  12. int main()  
  13. {  
  14.     a = 100;  
  15.     fun1(200, 300);  
  16.     fun2();  
  17.     fun2(100);  
  18.     return 0;  
  19. }  
  20. void fun2();
     

  4.我们从语法上来分析下#ifndef#define#endif(这点相信地球人都知道了)

  1. //-----a.h-----  
  2. #ifndef A_H_  
  3. #define A_H_  
  4. void fun();  
  5. #endif  

 

预编译阶段, 当第一次执行该段代码(即#include "a.h",参见第一条)时, 由于我们并没有宏定义A_H_, 所以会执行#define A_H_以及void fun()两条语句, 第二次执行该段代码时因为#ifndef A_H_为假就直接走到#endif后面也就等于该次#include "a.h"什么也没做了

总结:

好了, 下面就我们以上所学的知识来总结一下, 来纠正大家一直以来对#ifndef#define#endif的误解

当我们一个简单的project中有三个文件main.cpp, a.cpp, a.h,而 main.cpp 和a.cpp分别包含了a.h,
在编译阶段,两个编译单元是都会分别包含a.h的, 即使他们使用了#ifndef#define#endif, 这也是为什么当a.h被多个文件包含时我们不允许在a.h中定义变量及函数的原因,
因为在链接阶段会出现重定义。 但是在a.h中定义一个static变量却是允许的, 因为static变量是模块性作用域, 就这个例子来说, 若我们在a.h中写static int sss = 0;那么main.cpp与a.cpp使用的sss将为2个独立的sss.

那么是否#ifndef#define#endif就没用了呢, 大家可以想想, 当我们a.cpp中写了多个#include "a.h"时, 如果我们使用了#ifndef#define#endif那么预编译阶段就只会包含一个a.h中的内容到a.cpp中, 你也许会说, 有谁会傻到在a.cpp中写多个#include "a.h"呢, 那么请考虑稍微复杂点的情况,当我们main.cpp中包含了a.h和b.h,
而a.h中我们又包含了b.h, 那么如果我们使用了#ifndef#define#endif则main.obj只会包含一份b.h。

抱歉!评论已关闭.