c++中的宏定义

来源:互联网 发布:波士顿法律 知乎 编辑:程序博客网 时间:2024/05/18 03:47

一、不带参数的宏定义

宏定义又称为宏代换、宏替换,简称“宏”

格式:

#define 标识符 字符串

其中的标识符就是所谓的符号常量,也称为“宏名”

预处理(预编译)工作也叫做宏展开:将宏名替换为字符串。

掌握"宏"概念的关键是“换”。一切以换为前提、做任何事情之前先要换,准确理解之前就要“换”。

即在对相关命令或语句的含义和功能作具体分析之前就要换,“不管三七二十一,先换了再说”。

那么剩下的问题就简单了:

1 把谁换掉?2 换成什么?

#define PI 3.1415926

把程序中出现的PI全部换成3.1415926

li9_1.c

说明:(1)宏名一般用大写

(2)使用宏可提高程序的通用性和易读性,减少不一致性,减少输入错误和便于修改。

例如:数组大小常用宏定义

(3)预处理是在编译之前的处理,而编译工作的任务之一就是语法检查,预处理不做语法检查。

(4)宏定义末尾不加分号;

(5)宏定义写在函数的花括号外边,作用域为其后的程序,通常在文件的最开头。

(6)可以用#undef命令终止宏定义的作用域

(7)宏定义可以嵌套

li9_2.c

(8)字符串""中永远不包含宏

(9)宏定义不分配内存,变量定义分配内存。

二、带参数的宏

除了一般的字符串替换,还要做参数代换

格式:

#define 宏名(参数表) 字符串

例如:#define S(a,B) a*b

area=S(3,2);第一步被换为area=a*b;,第二步被换为area=3*2;

类似于函数调用,有一个哑实结合的过程

li9_3.c

(1)实参如果是表达式容易出问题

#define S® r*r

area=S(a+B);第一步换为area=r*r;,第二步被换为area=a+b*a+b;

正确的宏定义是#define S® ®*®

(2)宏名和参数的括号间不能有空格

(3)宏替换只作替换,不做计算,不做表达式求解

(4)函数调用在编译后程序运行时进行,并且分配内存。宏替换在编译前进行,不分配内存

(5)宏的哑实结合不存在类型,也没有类型转换。

(6)函数只有一个返回值,利用宏则可以设法得到多个值

li9_4.c

(7)宏展开使源程序变长,函数调用不会

(8)宏展开不占运行时间,只占编译时间,函数调用占运行时间(分配内存、保留现场、值传递、返回值)

三、预处理

预处理程序是一些行首以#开始的特殊语句,例如:#include,#define等就是预处理语句。在编译程序的编译过程中,进行其它编译处理(词法分析、语法分析、代码生成、优化和连接等)之前,先进行这些语句的分析处理。

预处理语句使用的目的在于帮助程序员编写出易读、易改、易移植并便于调试的程序。预处理语句主要有四种:

宏定义和宏替换、文件包含、条件编译和行控制。

预处理语句的作用范围是从被定义语句开始直至被解除定义或是到包含它的文件结术为止均有效。

四、宏定义和宏替换



“宏”是借用汇编语言中的概念。为的是在C语言程序中方便的作一些定义和扩展。这些语句以#define开头,分为两种:符号常量的宏定义和带参数的宏定义。

1.符号常量的宏定义和宏替换

符号常量的宏定义语句是一般格式:

#define 标识符 字符串

其中标识符就叫作宏名称。

注意:标识符与字符串之间不要用‘=’,结尾不要加‘;’。

2.带有参数的宏定义及其替换

复杂的宏定义带有参数列表,参数列表中可有不止一个参数,其一般格式:

#define 标识符(参数列表) 字符串

对带有参数的宏定义进行宏替换时,不仅对宏标识符作字符串替换,还必须作参数的替换。

例如:

#define SQ(x) ((x)*(x))

那么SQ(a+B)将被宏替换成(a+B)*(a+B)。

宏定义也可嵌套使用,即一个宏定义可用另一个宏定义来定义。例如:

#define SQ(x) ((x)*(x))

#define CUBE(x) (SQ(x)*(x))

3.宏定义类函数

宏定义常用于把直接插入的代码来代替函数,以提高执行效率。这一类的宏,就称做宏定义类函数,例如:

#define MIN(x,y) (((x)<(y))?(x):(y))

有了这样的宏之后,就可以直接引用,例如:

m=MIN(a,B);

这语句将被预处理成:

m=(((a)<(B))?(a):(B));