C语言一般提供三种预处理功能:宏处理、文件包含、条件编译。头文件防卫式申明中会用到条件编译中#ifndef
、#define
、#endif
的用法。所以,首先价绍下条件编译。
1 条件编译
一般情况下,在生成可执行文件的过程中,源程序文件中的所有代码行都进行编译,但是在一些跨操作系统的情况下,要求代码既能在Windows
下编译运行,也能在Linux
下编译运行,因为在不同的操作系统下调用的某些函数只能在特定的操作系统编译运行,此时就需要使用条件编译,让部分代码在满足特定条件下编译。
条件编译的几种格式
格式 1
程序段代码1#else 程序段代码2#endif#ifdef标识符
作用:当标识符被定义过,则对程序段代码 1 进行编译,否则对程序段 2 进行编译。
平时,在进行程序调试过程中,需要输出一些信息方便调试,在调试结束后,不需要这些信息输出,我们可以这样处理:
//然后在代码中需要输出调试信息的地方,写一些输出信息,例如:#ifdefDEBUGprintf("调试需要输出的提示信息\n");#endif#defineDEBUG
格式 2
程序段代码1#else 程序段代码2#endif#ifndef标识符
作用:若标识符未被定义,则编译程序段代码 1,否则编译程序段代码 2。与格式 1 正好相反,RELEASE
模式与DEBUG
模式正好相对,所以格式 1 调试的例子也可以写成这样:
//然后在代码中需要输出调试信息的地方,写一些输出信息,例如:#ifndefRELEASEprintf("调试需要输出的提示信息\n");#endif#defineRELEASE
格式 3
程序段代码1#elseif表达式2 程序段代码2#else 程序段代码3#endif#if表达式
跨平台项目开发,采用条件编译可以同同一套代码在不修改代码的情况下在Windows
平台与Linux
平台编译通过生成可执行文件,增加代码的可移植性。例如;
//Linux专有函数代码#elif_Win32 //windows专有函数代码#else //其他平台专有函数代码#endifif__Linux__
2 头文件防卫式声明
在多文件包含的情况下,有些变量何你可能被直接的或者间接的重复定义,重复#include
的问题也可能发生,可以通过#ifndef
、#define
、#endif
防卫式声明解决这一问题。
范例:
头文件 head1.h 有如下定义:
intg_head1=1;
头文件 head2.h 有如下定义:
intg_head2=2;
源文件 .cpp 中使用g_head1
、g_head2
代码如下:
usingnamespacestd; intmain() { cout<cout<return0; }#include"head1.h"#include"head2.h"#include
随着项目增大或者其他需求,可能出现头文件的包含,例如头文件head2.h
中包含头文件head1.h
,头文件head2.h
如下:
intg_head2=2;#include"head1.h";
此时编译就会出现重复定义的错误,这是因为源文件.cpp 包含了头文件head1.h
、head2.h"
,head2.h
中也包含head1.h
头文件,所以head1.h
定义的g_head1
被定义两次。
头文件head1.h
防卫式声明改造
intg_head1=1;#endif#ifndef_HEAD1_#define_HEAD1_
头文件head2.h
防卫式声明改造
intg_head2=2;#endif#ifndef_HEAD2_#define_HEAD2_#include"head1.h"
修改后再次编译,通过并成功执行,使用#ifndef
、#define
、#endif
组合的防卫式声明,避免了头文件内容被多次include
,所以在写 .h 文件时,要习惯性的使用文件防卫式声明。