C语言学习总结

关于#define的使用:

a.概念:在C或C++语言源程序中允许用一个标识符来表示一个字符串,称为“宏”。被定义为“宏”的标识符称为“宏名”。在编译预处理时,对程序中所有出现的“宏名”,都用宏定义中的字符串去代换,这称为“宏代换”或“宏展开”。宏定义是由源程序中的宏定义命令完成的。宏代换是由预处理程序自动完成的。程度上提高程序的运行效率。

b.宏的类型:在C或C++语言中,“宏”分为有参数和无参数两种:

无参宏定义:(即宏名后不带参数。)

其定义的一般形式为:

#define 标识符 字符串

其中各个名称的含义如下:
1>其中的“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。
2>“define”为宏定义命令。
3>“标识符”为所定义的宏名。
4>“字符串”可以是常数、表达式、格式串等。
例如:

 #define M (a+b)

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

程序1:

#define M (a+b)
 main(){    
 int s,y;     
 printf("input a number: ");   
 scanf("%d",&y);   
 s=M*M;     
 printf("s=%d\n",s);   
}   

上例程序中首先进行宏定义,定义M来替代表达式(a+b),在 s= M * M 中作了宏调用。在预处理时经宏展开后该语句变为:

S=(a+b)*(a+b)  

但要注意的是,在宏定义中表达式(a+b)两边的括号不能少。否则会发生错误。
如当作以下定义后:

#difine M (a)+(b)

在宏展开时将得到下述语句:

S= (a)+(b)*(a)+(b)  

对于宏定义还要说明以下几点:

1>宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。
2>宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。
3>宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。
4>如要终止其作用域可使用:

#undef命令 
带参宏定义:(即允许宏带有参数)

a.在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。
b.对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。

带参宏定义的一般形式为:

#define 宏名(形参表) 字符串(在字符串中含有各个形参) 

例如:

#define M(y) ((y)*(y)+3*(y)) /*宏定义*/ 
....  
k=M(5); /*宏调用*/ 
....     

在宏调用时,用实参5去代替形参y,经预处理宏展开后的语句为:

 k=5*5+3*5  

程序2:

#define MAX(a,b) (a>b)?a:b    
 main(){   
 int x,y,max;   
 printf("input two numbers: ");   
 scanf("%d%d",&x,&y);   
 max=MAX(x,y);     
 printf("max=%d\n",max);   
 }     

上例程序的第一行进行带参宏定义,用宏名MAX表示条件表达式:

(a>b)?a:b

形参a,b均出现在条件表达式中。
程序第七行

max=MAX(x,y)

为宏调用,实参x,y,将代换形参a,b。
宏展开后该语句为:

max=(x>y)?x:y;  

用于计算x,y中的大数。
注:
a.防止重复定义#define条件编译
b.头文件(.h)可以被头文件或C文件包含;
c.重复包含(重复定义)
d.由于头文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题的。 通过条件编译开关来避免重复包含(重复定义)
例如:

#ifndef __headerfileXXX__  
#define __headerfileXXX__ 
„ 
文件内容  
„       
#endif

使用中的常见问题解析

1.简单宏定义使用中出现的问题

在简单宏定义的使用中,当替换文本所表示的字符串为一个表达式时,容易引起误解和误用。
例如:

#define N 2+2
void main(){
   int a=N*N;
   printf(“%d”,a);
}
(1) 出现问题:

在此程序中存在着宏定义命令,宏N代表的字符串是2+2,在程序中有对宏N的使用,一般同学在读该程序时,容易产生的问题是先求解N为2+2=4,然后在程序中计算a时使用乘法,即N*N=4*4=16,其实该题的结果为8,为什么结果有这么大的偏差?

(2) 问题解析:

如1节所述,宏展开是在预处理阶段完成的,这个阶段把替换文本只是看作一个字符串,并不会有任何的计算发生,在展开时是在宏N出现的地方 只是简单地使用串2+2来代替N,并不会增添任何的符号,所以对该程序展开后的结果是a=2+2*2+2,计算后=8,这就是宏替换的实质,如何写程序才能完成结果为16的运算呢?

(3)解决办法:
/*将宏定义写成如下形式*/
#define N (2+2)
/*这样就可替换成(2+2)*(2+2)=16*/

2.带参数的宏定义出现的问题

在带参数的宏定义的使用中,极易引起误解。例如我们需要做个宏替换能求任何数的平方,这就需要使用参数,以便在程序中用实际参数来替换宏定义中的参数。
一般学生容易写成如下形式:

#define area(x) x*x
/*这在使用中是很容易出现问题的,看如下的程序*/

void main()
{
    int y = area(2+2);
    printf(“%d”,y);
}
(1) 问题解析:

按理说给的参数是2+2,所得的结果应该为4*4=16,但是错了,因为该程序的实际结果为8,仍然是没能遵循纯粹的简单替换的规则,又是先计算再替换 了,在这道程序里,2+2即为area宏中的参数,应该由它来替换宏定义中的x,即替换成2+2*2+2=8了。
那如果遵循(1)中的解决办法,把2+2 括起来,即把宏体中的x括起来,是否可以呢?#define area(x) (x)(x),对于area(2+2),替换为(2+2)(2+2)=16,可以解决,但是对于area(2+2)/area(2+2)又会怎么样呢,有的学生一看到这道题马上给出结果,因为分子分母一样,又错了,还是忘了遵循先替换再计算的规则了,这道题替换后会变为 (2+2)(2+2)/(2+2)(2+2)即4*4/4*4按照乘除运算规则,结果为16/4*4=4*4=16,那应该怎么呢?

(2)解决方法:

在整个宏体上再加一个括号,即#define area(x) ((x)*(x)),不要觉得这没必要,没有它,是不行的。

要想能够真正使用好宏定义,一定要先将程序中对宏的使用全部替换成它所代表的字符串,完全展开后再进行相应的计算,就不会写错运行结果。如果是自己编程使用宏替换,则在使用简单宏定义时,当字符串中不只一个符号时,加上括号表现出优先级,如果是带参数的宏定义,则要给宏体中的每个参数加上括号,并在整个宏体上再加一个括号。看到这里,不禁要问,用宏定义这么麻烦,这么容易出错,可不可以摒弃它, 那让我们来看一下在C语言中用宏定义的好处吧。
例如:

#include <iostream.h>
#define product(x)    x*x
int main()
{
    int i=3;
    int j,k;
    j = product(i++);
    cout<<"j="<<j<<endl;
    cout<<"i="<<i<<endl;
    k = product(++i);
    cout<<"k="<<k<<endl;
    cout<<"i="<<i<<endl;
    return 0;
}

依次输出结果:
j=9;i=5;k=49;i=7

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值