zoukankan      html  css  js  c++  java
  • C语言中带参数的宏

    带参数的宏定义有如下的格式:

    【#define 指令----带参数的宏】 #define 标识符(x1,x2,……,xn)

    其中  x1,x2,……xn是标志符(宏的参数)

    注意:在宏的名字和括号之间必修没有空格。

    如果有空格,预处理会认为是在定义一个简单的宏,其中(x1,x2,……,xn)是替换列表的一部分

    当预处理器遇到一个带参数的宏,会将定义存储起来以便以后使用。在后面的程序中,如果任何地方出现了标识符(y1,y2……,yn)格式的宏调用(其中y1,y2, ……yn是一些列标记),预处理器会使用替换列表替代,并使用yi替换xi

    e.g. 假如我们定义了如下的宏:

    #define MAX(x,y)  ((x)>(y) ? (x) : (y))
    
    #define IS_EVEN(n)  ((n)%2==0)

    下面的例子是一个更加复杂的宏:

    #define TOUPPER(c)  ('a'<=(c) && (c)<='z' ? (c)-'a'+'A' : (c))

    带参数的宏可以包含空的参数列表,如下所示:

    #define getchar() getc(stdin)

    空的参数列表不是一定确实需要,但可以使getchar更像一个函数

    使用带参数的宏替代实际的函数有两个优点:

    • 程序可能会稍微快些。一个函数调用在执行时通常会有些额外开销----存储上下文信息、复制参数的值等。而一个宏的调用则没有这些运行开销
    • 宏会更“通用”。与函数的参数不同,宏的参数没有类型。因此,只要预处理后的程序依然合法的,宏可以接受任何类型的参数。 e.g.我们可以使用MAX宏从两个数中选出较大的一个,数的类型可以是:int,long int,float,double等等
    • 编译后的代码通常会变大。每一处宏调用都会导致插入宏的替换列表,由此导致程序源代码增加(因此编译后的代码变大)。宏使用得越频繁,这种效果就越明显。当宏调用嵌套时,这个问题会相互叠加从而使程序更加复杂。e.g. n=MAX(i,MAX(j,k));

    下面是预处理后的这条语句:      

    n=((i)>(((j)>(k)?(j):(k)))?(i):(((j)>(k)?(j):(k))));
    • 宏参数没有类型检查。
    • 无法用一个指针来指向一个宏
    • 宏可能会不止一次地计算它的参数。函数对它的参数只会计算一次,而宏可能会计算两次甚至更多。如果参数有副作用,多次计算参数的值可能会产生意外的结果

    考虑下面的例子,其中MAX的一个参数有副作用:

    n=MAX(i++, j)

    下面是这条语句在预处理之后的结果:

    n=((i++) >(j)?(i++):(j));

    带参数的宏不仅适用于模拟函数的调用,他们特别经常被作为模板,来处理我们经常要重复书写代码段:

    e.g.我们可以使用

    #define PRINT_INT(x)  printf("%d\n",x);

    使得PRINT_INT(x)代替每次使用的printf("%d\n",x);

  • 相关阅读:
    linux ubuntu装机到可实现java(eclipse,intellij IDEA,android)开发全过程
    浅谈线程同步的几种方法
    KMP算法,这是我看到的最简单的最好理解的KMP算法
    常用基础算法C++实现
    堆内存和栈内存详解(转载)
    数据结构=。= 链表
    倒排索引--资料1
    倒排索引简单理解
    简单理解Socket
    8.结构体的使用 2015.12.3
  • 原文地址:https://www.cnblogs.com/cpoint/p/3367386.html
Copyright © 2011-2022 走看看