1,#define 叫做宏定义命令,它也是C语言预处理命令的一种,所谓宏定义,就是用一个标识符来表示一个字符串,如果在后面的代码中出现了该标识符,那么就全部替换成指定的字符串
2,
#define N 100 就是宏定义,N为宏名,100是宏的内容(宏所表示的字符串)。在预处理阶段,对程序中所有出现的“宏名” ,预处理器都会用宏定义中的字符串去替换,这称为“宏替换” 或 “宏展开”。 宏定义是由源程序中的宏定义命令 #define 完成的,宏替换是由预处理程序完成的。
3,宏定义的形式
#define 宏名 字符串
①#表示这是一条预处理命令,所有的预处理命令都以 # 开头,宏名是标识符的一种,命名规则和变量相同,字符串可以是数字,表达式,if语句,函数等。
②这里所说的字符串是一般意义上的字符序列,不要和C语言中的字符串等同,它不需要双引号。
③程序中反复使用的表达式就可以使用宏定义
4,宏定义注意事项
①宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的替换,字符串中可以含任何字符,它可以是常数,表达式,if语句,函数等。预处理程序对它不作任何检查,如有错误,只能在编译已被宏展开后的源程序时发现。
②宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起替换。
③宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束,如要终止其作用域可使用 #undef 命令。
④代码中的宏名如果被引号包围,那么预处理程序不对其做宏代替
⑤宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名,在宏展开时由预处理程序层层代换。
⑥习惯上宏名用大写表示,以便于与变量区别,但也允许用小写字母
⑦可用宏定义表示数据类型,使书写方便
#define UINT unsigned int
void main(){
UINT a, b ; //宏替换 unsigned int a,b
}
⑧宏定义表示数据类型时和用 typeof 定义数据类型的区别:
宏定义只是简单的字符串替换,由预处理器来处理,而typeof 是在编译阶段由编译器处理的,它并不是简单的字符串替换,而是给原有的数据类型起一个新的名字,将它作为一 种新的数据类型
5,带参数的宏定义
①C语言允许宏带有参数,在宏定义中的参数称为“形式参数”,在宏调用中的参数称为“实际参数”,这点和函数有些类似
②对带参数的宏,在展开过程中不仅要进行字符串替换,还有用实参去替换形参
③带参宏定义的一般形式为 #define 宏名(形参列表)字符串,在字符串中可以含有各个形参
④带参宏调用的一般形式为 宏名(实参列表)
⑥带参宏定义中,形参之间可以出现空格,但是宏名和形参列表之间不能有空格
#define MAX(a,b) (a>b) ? a: b 如果写成 #define MAX (a , b) (a>b) ? a: b
将被认为是无参宏定义,宏名 MAX 代表字符串 (a , b) (a>b) ? a: b
⑦在带参宏定义中,不会为形式参数分配内存,因此不必指明数据类型,而在宏调用中,实参包含了具体的数据,要用它们去替换形参,因此实参必须要指明数据类型。
⑧在宏定义中,字符串内的形参通常要用括号括起来以避免出错。
6,带参宏定义和函数的区别
①宏展开仅仅是字符串的替换,不会对表达式进行计算,宏在编译之间就被处理掉了,它没有机会参与编译,也不会占用内存
②函数是一段可以重复使用的代码,会被编译,会给它分配内存,每次调用函数,就是执行这块内存中的代码
7,常见的预处理指令
# 空指令,无任何效果
#include 包含一个源代码文件
#define 定义宏
#undef 取消已定义的宏
#if 如果给定条件为真,则编译下面代码
#ifdef 如果宏已经定义,则编译下面代码
#ifndef 如果宏没有定义,则编译下面代码
#elif 如果前面给定的#if 条件不为真,当前条件为真,则编译下面代码
#endif 结束一个 #if ... #elif 条件编译块