预处理器是一些指令,指示编译器在实际编译之前所需完成的预处理。
预处理器的主要作用就是: 把通过预处理的内建功能对一个资源进行等价替换,最常见的预处理有: 文件包含,条件编译、布局控制和宏替换4种。
文件包含: #include 是一种最为常见的预处理,主要是做为文件的引用组合源程序正文。
条件编译: #if,#ifndef,#ifdef,#endif,#undef等也是比较常见的预处理,主要是进行编译时进行有选择的挑选,注释掉一些指定的代码,以达到版本控制、防止对文件重复包含的功能。
布局控制: #progma,这也是我们应用预处理的一个重要方面,主要功能是为编译程序提供非常规的控制流信息。
宏替换: #define,这是最常见的用法,它可以定义符号常量、函数功能、重新命名、字符串的拼接等各种功能。
所有的预处理器指令都是以井号(#)开头,只有空格字符可以出现在预处理指令之前。预处理指令不是 C++ 语句,所以它们不会以分号(;)结尾。
C++ 还支持很多预处理指令,比如 #include、#define、#if、#else、#line 等。
常见的预处理指令:
#define 宏定义
#undef 取消宏
#include 文本包含
#ifdef 如果宏被定义就进行编译
#ifndef 如果宏未被定义就进行编译
#endif 结束编译块的控制
#if 表达式非零就对代码进行编译
#else 作为其他预处理的剩余选项进行编译
#elif 这是一种#else和#if的组合选项
#line 改变当前的行数和文件名称
#error 输出一个错误信息
#pragma 为编译程序提供非常规的控制流信息
#define 预处理
#define 预处理指令用于创建符号常量。该符号常量通常称为宏,指令的一般形式是:
#define name data
name:通常大写,data:数字或字符
参数宏
可以使用 #define 来定义一个带有参数的宏
#include<iostream>
using namespace std;
#define MUL1(a,b) a*b
#define MUL2(a,b) (a)*(b)
int main()
{
cout << MUL1(1+2, 2+3) << endl;
cout << MUL2(1+2, 2+3) << endl;
system("pause");
return 0;
}
条件编译
有几个指令可以用来有选择地对部分程序源代码进行编译。这个过程被称为条件编译。
条件预处理器的结构与 if 选择结构很像。
//如果没有定义TEN,则重新定义TEN
#ifndef TEN
#define TEN 10
#endif
可以只在调试时进行编译,调试开关可以使用一个宏来实现
如果在指令 #ifdef DEBUG 之前已经定义了符号常量 DEBUG,则会对程序中的 cerr 语句进行编译。
可以使用 #if 0 语句注释掉程序的一部分。
#include<iostream>
using namespace std;
#define MUL1(a,b) a*b
#define MUL2(a,b) (a)*(b)
//如果没有定义TEN,则重新定义TEN
#ifndef TEN
#define TEN 10
#endif
#define ONE
int main()
{
cout << MUL1(1+2, 2+3) << endl;
cout << MUL2(1+2, 2+3) << endl;
cout << TEN << endl;
int x = 9;
#ifdef ONE
cerr << "x=" << x << endl;
#endif
#if 0
/* 这是注释部分 */
cout << MKSTR(HELLO C++) << endl;
#endif
system("pause");
return 0;
}
# 和 ## 运算符
# 和 ## 预处理运算符在 C++ 和 ANSI/ISO C 中都是可用的。# 运算符会把 replacement-text 令牌转换为用引号引起来的字符串。
#include<iostream>
using namespace std;
#define STRING(a) #a
int main()
{
cout << STRING(你好!) << endl;
system("pause");
return 0;
}
C++将
cout << STRING(你好!) << endl;
转换成:
cout << "你好! "<< endl;
## 运算符用于连接两个令牌。
#define STRING(a,b) a##b
#include<iostream>
using namespace std;
#define STRING1(a,b) #a#b
#define STRING(x,y) x##y
int main()
{
string xy1 = "HelloWorld";
int xy = 100;
cout << STRING(x,y) << endl;
//cout << STRING(x, y1) << endl;//字符串不能
cout << STRING1(Hello,World) << endl;
system("pause");
return 0;
}
C++ 中的预定义宏
C++ 提供了下表所示的一些预定义宏:
宏 | 描述 |
---|---|
__LINE__ | 这会在程序编译时包含当前行号。 |
__FILE__ | 这会在程序编译时包含当前文件名。 |
__DATE__ | 这会包含一个形式为 month/day/year 的字符串,它表示把源文件转换为目标代码的日期。 |
__TIME__ | 这会包含一个形式为 hour:minute:second 的字符串,它表示程序被编译的时间。 |
#include <iostream>
using namespace std;
int main ()
{
cout << "Value of __LINE__ : " << __LINE__ << endl;
cout << "Value of __FILE__ : " << __FILE__ << endl;
cout << "Value of __DATE__ : " << __DATE__ << endl;
cout << "Value of __TIME__ : " << __TIME__ << endl;
return 0;
}
待续。。。