预处理命令与位运算

本文详细介绍了C语言中的预处理命令,特别是宏定义,包括无参宏和带参宏的使用方法、注意事项以及与函数的区别。通过实例展示了宏定义如何在预处理阶段进行字符串替换,并提醒读者在使用宏定义时需要注意的潜在问题,以避免错误。
摘要由CSDN通过智能技术生成

  所谓预处理是指在进行编译的第一遍(词法扫描和语法分析)之前所做的工作。预处理是C语言的一个重要功能,它由预处理程序负责完成。当对一个原文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分进行处理,处理完毕自动进入对源程序的编译。

  第一部分 宏定义

  在C语言源程序中允许用一个标识符来表示一个字符串,成为“宏”。被定义为“宏”的标识符成为“宏名”。在编译处理命令时,对程序中所出现的“宏名”,都是宏定义中的字符串去代替,这称为“宏代换”或“宏展开”。

  宏定义是由源程序中的宏定义命令完成的。宏代换是由预处理程序自动完成的。

  在C语言中,“宏”分为有参数和无参数两种。

  1.1  无参宏定义

   无参宏的宏名不带参数。其定义的一般形式为:

   #define 标识符 字符串

  其中“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define“为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常量、表达式或格式串等。

  在前面介绍过的符号常量的定义就是一种无参宏定义。此外,对程序中反复使用的表达式也可进行宏定义。如:#define M (y*y+3*y)

它的作用是指定标识符M来代替表达式(y*y+3*y)。在编写源程序时,所有的(y*y+3*y)都可由M代替,而对源程序进行编译时,将先由预处理程序进行宏代换,即用(y*y+3*y)表达式去置换所有的宏名M,然后再进行编译。

例1:红的定义和替换

#include<stdio.h>
#define M (y*y+3*y)
main(){
 int s,y;
 printf("input a number:/n");
 scanf("%d",&y);
 s=3*M+4*M+5*M;
 printf("s=%d/n",s);
}

  首先定义了宏,定义M来替换表达式(y*y+3*y),在s=3*M+4*M+5*M;中进行宏调用。在预处理时经宏展开后该语句变为:s=3*
(y*y+3*y)+4*(y*y+3*y)+5*(y*y+3*y);但是要注意在宏定义中表达式(y*y+3*y)两边的括号不能少。否则会产生错误。

  对于宏定义有几点说明:

(1)宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以包含任何字符字符,可以是常量,也可以是表达式,预处理程序对他不做任何检查。如有错误,只能在编译已被展开后的源程序时发现。

(2)宏定义不是类型声明或语句,在行末不必加分号,如加分号则连分号也一起置换。

(3)宏定义必须写在函数之外,其作用于从宏定义命令起到源程序结束。如要终止起作用域也是用#undef命令。

(4)宏名在源程序中若用引号括起来,则预处理程序不对其进行宏代换。

(5)宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名。在宏展开时由预处理程序层层代换。

(6)习惯上宏名用大写字符表示,以便与变量区分。但也允许用小写字母。

(7)可用宏定义表示数据类型,使书写方便。

如:

   #define STU struct stu

 在程序中可用STU做变量说明:STU body[5],*p;

  #define INTEGER int

 在程序中就可用INTEGER进行整形变量说明:INTEGER a,b;

 应注意用宏定义表示数据类型和typedef定义数据说明符的区别。宏定义只是简单的字符串替换,是在预处理阶段完成的,而typedef是在编译阶段进行处理的,他不是做简单的代换,而是对类型说明符重新命名。被命名的标识符具有类型定义说明的功能。

  请看下面的例子:

   #define PIN1 int *

   typedef (int *)PIN2;

 从形式上看这两者相似,但在实际使用中却不相同。下面用PIN1,PIN2声明变量就可以看出他们的区别:

 PIN1 a,b;在宏替换后变成: int * a,b;表示a是指向整形的指针变量,而b是整型变量。然而:PIN2 a,b;表示a,b都是指向整形的指针变量。因为PIN2是一个类型说明符。由此例可知,宏定义虽然也可表示数据类型,但毕竟是做字符替换。在使用时要格外小心,避免出错。

例3:对“输入格式”做宏定义

#include<stdio.h>
#define P printf
#define D "%d,"
#define F "%f/n"
main(){
 int a=5,c=8,e=11;
 float b=3.8,d=8.7,f=21.08;
 P(D F,a,b);
 P(D F,c,d);
 P(D F,e,f);
}

1.2 带参宏定义
 C语言允许宏带有参数。在宏定义中的参数成为形式参数,在宏调用中的参数成为实际参数。对于带参数的宏,在调用中,不仅要进行宏展开,而且要用实参去替换形参。

 带参宏定义的一般形式为:#define 宏名(形参表) 字符串

 在字符串中含有各个形参。带参宏调用的一般格式为:宏名(实参表);

如:#define M(y) y*y+3*y ... k=M(5); ...

例4:带参数的宏

#include<stdio.h>
#define MAX(a,b) (a>b)?a:b

main(){
 int x,y,max;
 printf("input two numbers:/n");
 scanf("%d%d",&x,&y);
 max=MAX(x,y);
 printf("max=%d/n",max);
}
 对带参的宏定义有以下几点说明:

(1)带参宏定义中,宏名和形参表之间不能有空格。

(2)在带参宏定义中,形式参数不分配内存空间,因此不必做类型定义。而宏调用中的实参有具体的值。要用它们去替换形参,因此必须做类型说明。这是与函数中的情况不同的。在函数中,形参和实参是两个不同的量,各有自己的作用域,调用时要把实参值赋值予形参,进行“值传递”。而在带参宏中,只进行符号替换,不存在值传递的问题。

(3)在宏定义中的形参是标识符,而宏调用中的实参可以是表达式。

(4)宏定义中,字符串内的形参通常把实参表达式的值用括号括起来以避免出错。

(5)带参的宏和带参的函数很相似,但本质上是不同的,除上面已谈到的几点外,把同一表达式用函数处理与用宏处理后,两者的结果可能是不同的。

例5:带参的函数

#include<stdio.h>
main(){
 int i=1;
 while(i<5)
  printf("%d  ",SQ(i++));
}
SQ(int y){
 return (y*y);
}

例6:带参的宏

#include<stdio.h>
#define SQ(y) (y*y)
main(){
 int i=1;
 while(i<=5)
  printf("%d  ",SQ(i++));
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值