在预处理之前,编译器必须对程序进行一些翻译处理
编译器把源代码中出现的字符映射到源字符集
编译器定位每个反斜杠后面跟着换行符的实例,并删除它们(可以把两个物理行转化为一个逻辑行)
编译器把文本划分为预处理记号序列、空白序列和注释序列(编译器将用一个空格字符替换每一条注释)
宏
-
预处理器发现程序中的宏后,会用宏等价的替换文本进行替换。如果替换的字符串中还包含宏,则继续替换这些宏,唯一例外的是双引号中的宏。
-
类函数宏与函数(宏和函数的选择是时间和空间的权衡)
宏调用是字符串替换,在编译之前把参数记号传递给程序(一般不要再宏中使用递增或传递运算符) 函数调用是在程序运行时吧参数的值传递给函数 发生在不同的时期
-
宏的优点
优点:宏不用担心变量类型,只是字符串的替换 缺点:比普通函数复杂,稍有不慎就会产生奇怪的副作用
-
使用宏的注意点
宏名中不可以有空格,但是替换字符串可以有 用圆括号把宏的参数和整个替换括起来,能确保被括起来的部分在表达式中被正确地展开 如果在程序中只是用一次的宏无法明显减少程序的运行时间
例题
写一个比较标准的宏,这个宏输入两个参数并返回比较小的一个。
#define MIN(x,y) ((x) > (y) ? (x) : (y))
预处理
-
什么是预编译?何时需要预编译?
预编译又称预处理,是整个编译过程中最先做的工作,即程序执行前的一些预处理工作。 主要处理#开头的命令,比如靠背#include包含的文件代码、替换#define定义的宏、条件 编译#if等 总是使用不经常改动的大型代码体。 程序由多个模块组成,所有模块都是用一组标准的包含文件和相同的编译选项。在这种 情况下,可以将所有包含文件预编译为一个预编译头。
-
#与##的作用?
C允许在字符串中包含宏参数。#作为一个预处理运算符,可以把记号转化为字符串,这 个过程为字符串化 预处理器粘合剂:##运算符,它可以把两个记号组合成一个记号,可用于类函数宏的替 换部分,还可以用做对象宏的替换部分
#的作用
#include <stdio.h>
#define ADD1(A,B) printf("A + B = %d\n",((A) + (B)));
#define ADD2(A,B) printf(#A"+"#B "= %d\n",((A) + (B)));
int main()
{
ADD1(5.20);
ADD2(5,20);
return 0;
}
输出
A + B = 25
5 + 20 = 25
##的作用
#include <stdio.h>
#define XNAME(n) x##n
#define PRINT_XN(n) printf("x" #n "= %d\n",x ## n);
int main()
{
int XNAME(1) = 14;
int XNAME(2) = 20;
int x3 = 30;
PRINT_XN(1);+
PRINT_XN(2);
PRINT_XN(3);
return 0;
}
输出
x1 = 14
x2 = 20
x3 = 30
##运算符在这里看起来没有多大便利,但是在有的地方,是可以提高封装性以及程序的可读性的。比如STM32宏定义里,可以很方便的对GPIO口的操作进行封装。
#define __STM32_PIN(index, gpio, gpio_index) \
{ \
index, GPIO##gpio##_CLK_ENABLE, GPIO##gpio, GPIO_PIN_##gpio_index \
}