带参数的宏定义
基本介绍:
(1)C语言允许宏带有参数。在宏定义中的参数称为“形式参数”,在宏调用中的参数称为“实际参数”。
(2)对带参数的宏,在展开过程中不仅要进行字符串替换,还要用实参去替换形参。
(3)带参宏定义的一般形式为 #define 宏名(形参列表) 字符串,在字符串中可以含有各个形参。
(4)带参宏调用的一般形式为:宏名(实参列表); 。
#include<stdio.h>
#define MAX(a,b) (a>b)?a:b // MAX 带参数的宏,(a,b)行参
// a>b?a:b 字符串(三元表达式)
void main(){
int x,y,max;
printf("input two numbers: ");
scanf("%d %d",&x,&y);
max=MAX(x,y); // 调用带参数的宏定义
printf("max=%d\n",max);
}
==#define MAX(a,b) (a>b)?a:b 强烈建议改成 #define MAX(a,b) ((a)>(b))?(a):(b)防止边界不清导致错误。 ==
带参数宏定义的注意事项和细节
(1)带参宏定义中,形参之间可以出现空格,但是宏名和形参之间不能有空格。
#define MAX(a,b) (a>b)?a:b
MAX和(a,b)之间不能有空格
(a,b)和 a>b?a:b 之间可以有空格
(2)在带参宏定义中,不会为形式参数分配内存,因此不必指明数据类型。而在宏调用中,实参包含了具体的数据,要用它们去替换形参,因此实参必须要指明数据类型。
3)在宏定义中,字符串内的形参通常要用括号括起来以避免出错。
案例:计算 (10+1)^2 ,用宏定义
字符串中参数不带()
#include<stdio.h>
#define SQ(y) y*y
void main(){
int a,sum;
printf("input a number: ");
scanf("%d",&a);
sum = SQ(a+1); // 替换成 a + 1 * a + 1
printf("sum=%d\n",sum);
system("pause"); //程序暂停一下,然后按任意键继续
}
字符串中参数不带(),边界不清,导致错误。
字符串中参数带()
#include<stdio.h>
#define SQ(y) (y)*(y)
void main(){
int a,sum;
printf("input a number: ");
scanf("%d",&a);
sum = SQ(a+1); // 替换成 (a + 1) * (a + 1)
printf("sum=%d\n",sum);
system("pause"); //程序暂停一下,然后按任意键继续
}
带参数的宏定义和函数的区别
(1)宏定义仅仅是字符串的替换,不会对表达式进行计算;宏在编译之前就被处理掉了,它没有机会参与编译,也
不会占用内存。
(2)函数是一段可以重复使用的代码,会被编译,会给它分配内存,每次调用函数,就是执行这块内存中的代码。
案例说明:要求使用函数计算平方值,使用宏计算平方值,并总结二者的区别。
函数
#include<stdio.h>
#include<stdlib.h>
int SQ(int y){
return ((y)*(y));
}
void main(){
int i=1;
while(i<=5){
printf("%d^2 =%d\n",i-1,SQ(i++));
//printf() 函数是从后向前执行
}
}
分别计算了1到5的平方
带参数宏定义
#include<stdio.h>
#define SQ(y) ((y)*(y))
void main(){
int i=1;
while(i<=5){
printf("%d^2 =%d\n",i-2,SQ(i++));//SQ(i++)展开后 (i++)*(i++),i加了2次
}
}
分别计算了1,3,5的平方
常用的预处理指令总结
指令 | 说明 |
---|---|
# | 空指令,无任何效果 |
#include | 包含一个源代码文件 |
#define | 宏定义 |
#undef | 取消已定义的宏 |
#if | 如果给定条件为真,则编译下面代码 |
#ifdef | 如果宏已经定义,则编译下面代码 |
#ifndef | 如果宏没有定义,则编译下面代码 |
#elif | 如果前面的if给定条件不为真,当前条件为真,则编译下面代码 |
#endif | 结束一个#if…#else条件编译块 |
预处理指令使用注意事项
(1)预处理功能是C语言特有的功能,它是在在源程序正式编译前由预处理程序完成的,程序员在程序中用预处理
命令来调用这些功能。
(2)宏定义可以带有参数,宏调用时是以实参代换形参,而不是“值传送”。
(3)为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符串中出现的形式参数两边也应加括号。
(4)文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标
文件。
(5)条件编译允许只编译源程序中满足条件的程序段,使生成的目标程序较短,从而减少了内存的开销并提高了程
序的效率。
(6)使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。