#ifdef和#ifndef,这是条件编译的宏定义。
一般程序在调试过程的编译跟正式发行的时候是不一样的,调试过程中可能有很多测试语句是用户不需要的,但是如果一行行删掉或者注释掉又很麻烦,并且如果发行之后又需要修改BUG又要用到之前的测试,重写也是很麻烦的。于是可以考虑用条件编译。
首先可以写一个宏定义来表示是否按调试模式编译
#define DEBUG
然后需要写测试语句的时候都这样写
#ifdef DEBUG
测试语句
#endif
意思是,如果DEBUG已经被定义,则编译时将以下语句编译,否则忽略,这样,只需要把#define DEBUG这一句注释掉,编译器就可以忽略所有测试语句,而只要去掉注释,编译器又会把这些测试语句编译,非常方便。
#ifndef类似,就是如果此标识符没有被定义过,则编译下列语句。


一般情况下,源程序中所有的行都参加编译。但是有时希望对其中一部分内容只在满足一定条件才进行编译,也就是对一部分内容指定编译的条件,这就是“条件编译”。有时,希望当满足某条件时对一组语句进行编译,而当条件不满足时则编译另一组语句。

条件编译命令最常见的形式为:
#ifdef 标识符
程序段1
#else
程序段2
#endif

它的作用是:当标识符已经被定义过(一般是用#define命令定义),则对程序段1进行编译,否则编译程序段2。
其中#else部分也可以没有,即:
#ifdef
程序段1
#denif

在头文件中使用#ifdef和#ifndef是非常重要的,可以防止双重定义的错误。如你在头文件aaa.h中定义了一个类aaa如下:
class aaa
{
};
如果两次#include “aaa.h”(不见得是直接,也有可能两个不同的头文件中都包含了这个头文件)就会出错,因为相同的类不能定义两次。把aaa.h稍做修改:
#ifndef aaa
#define aaa
class aaa
{
};
#endif
就可以避免这样的问题。因为当你已经包含过这个文件,_aaa_就会有了定义,那么#ifndef的条件为假,就不会再执行后面的类定义了。

#ifdef和#endif必须成对使用。
从理论上讲可以出现在任何地方(头文件和实现文件中)
通常为了防止头文件被多次包含,在头文件中使用是必须的:
如:#ifndef MY_HEAD_H //头文件开头,名字是任意的,注意不要和其它头文件冲突

头文件声明
#endif //头文件结尾

有时候,在b.h中会include “a.h” ,在"c.h"中会include “b.h"及include"a.h”, 这时,如果不用ifndef/endif,就会包含两次a.h,产生错误。

还是把头文件的内容都放在#ifndef和#endif中吧。不管你的头文件会不会被多个文件引用,你都要加上这个。一般格式是这样的:

#ifndef <标识>
#define <标识>


#endif

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

#ifndef STDIO_H
#define STDIO_H

#endif