编译预处理
专题三:编译预处理。包括以下章节:
- 编译过程简介
- 宏定义与宏使用分析
- 条件编译使用分析
- #error和#line
- #pragma预处理分析
- #和##运算符使用解析
基本概念
- 条件编译的行为类似于c语言中的if…else…语句
- 条件编译是预编译指示命令,用于控制编译哪一段代码
条件编译对预编译器起作用;if…else…在运行时起作用
#if()…#else…#endif 判断条件表达式
- #ifdef…#else…#endif 判断宏
- #ifndef…#endif 判断是否定义过宏:如果定义过,跳出;否则,往下进行
3-1.c
#include<stdio.h>
//如果没有定义宏常量C,可以通过命令:gcc -DC=1 -E 3-1.c -o 3-1.i 来定义宏常量C,方便调试。
#define C 1
int main()
{
#if( C == 1)
printf("THIS IS FIRST ...\n");
#else
printf("THIIS IS SECOND ...\n");
#endif
if (c == 1) {
printf("THIS IS FIRST ...\n");
}else {
printf("THIIS IS SECOND ...\n");
}
return 0;
}
3-1.i
# 1 "3-1.c"
# 1 "<built-in>"
# 1 "<命令行>"
# 1 "3-1.c"
/*......此处省略n行......*/
# 940 "/usr/include/stdio.h" 3 4
# 2 "3-1.c" 2
int main()
{
printf("THIS IS FIRST ...\n");
if (1 == 1) {
printf("THIS IS FIRST ...\n");
}else {
printf("THIIS IS SECOND ...\n");
}
return 0;
}
#include
- #include是将已经存在的文件内容拷贝到当前文件中
- 间接包含同一个文件也会拷贝相同的内容到当前文件中,会出现编译错误
3-2.c
#include "3-2.h"
#include "global.h"
int main()
{
return 0;
}
3-2.h
#include "global.h"
global.c
int global = 2;
3-2.i
# 1 "3-2.c"
# 1 "<built-in>"
# 1 "<命令行>"
# 1 "3-2.c"
# 1 "3-2.h" 1
# 1 "global.h" 1
int global = 2; //重定义了global
# 2 "3-2.h" 2
# 2 "3-2.c" 2
# 1 "global.h" 1
int global = 2;
# 3 "3-2.c" 2
int main()
{
return 0;
}
结果:
- 我们可以使用条件预编译指令(#ifndef:判断是否定义过宏; #define:定义宏定义; #endif:结束)解决嵌套包含问题
3-2.c
#include "3-2.h"
#include "3-2.h"
#include "global.h"
#include "global.h"
int main()
{
return 0;
}
3-2.h
#ifndef _3_2_H_
#define _3_2_H_
#include "global.h"
#endif
global.c
#ifndef _GLOBAL_H_
#define _GLOBAL_H_
int global = 2;
#endif
3-2.i
# 1 "3-2.c"
# 1 "<built-in>"
# 1 "<命令行>"
# 1 "3-2.c"
# 1 "3-2.h" 1
# 1 "global.h" 1
int global = 2;
# 5 "3-2.h" 2
# 2 "3-2.c" 2
int main()
{
return 0;
}
结果编译正确:
条件编译的意义
- 条件编译使得我们可以按不同的条件编译不同的代码段, 因而可以产生不同的目标代码
- #if…#else…#endif被预编译器处理;而if…else语句被 编译器处理,必然被编译进目标代码
- 实际工程中条件编译主要用于一下两种情况:
- 不同的产品线共用一份代码
- 区分编译产品的调试版和发布版
实例分析3-2:条件编译的使用
3-3.c
#include <stdio.h>
#ifdef DEBUG
#define LOG(s) printf("[%s:%d] %s\n", __FILE__, __LINE__, s)
#else
#define LOG(s) NULL
#endif
#ifdef HIGH
void f()
{
printf("This is the high level product!\n");
}
#else
void f()
{
}
#endif
int main()
{
LOG("Enter main() ...");
f();
printf("1. Query Information.\n");
printf("2. Record Information.\n");
printf("3. Delete Information.\n");
#ifdef HIGH
printf("4. High Level Query.\n");
printf("5. Mannul Service.\n");
printf("6. Exit.\n");
#else
printf("4. Exit.\n");
#endif
LOG("Exit main() ...");
return 0;
}
结果:
小结
- 通过编译器命令行能够定义预处理器使用的宏
- 条件编译可以避免重复包含头同一个头文件
- 条件编译是在工程开发中可以区别不同产品线的代码
- 条件编译可以定义产品的发布版和调试版