带参数的宏

  • C++提供的编译预处理功能主要有以下三种:
      (一) 宏定义
      (二) 文件包含
      (三) 条件编译
      在C++中,我们一般用const定义符号常量。很显然,用const定义常量比用define定义常量更好。
      在使用宏定义时应注意的是:
      (a) 在书写#define 命令时,注意<宏名>和<字符串>之间用空格分开,而不是用等号连接。
      (b) 使用#define定义的标识符不是变量,它只用作宏替换,因此不占有内存。
      (c) 习惯上用大写字母表示<宏名>,这只是一种习惯的约定,其目的是为了与变量名区分,因为变量名
      通常用小写字母。
      如果某一个标识符被定义为宏名后,在取消该宏定义之前,不允许重新对它进行宏定义。取消宏定义使用如下命令:
      #undef<标识符>
      其中,undef是关键字。该命令的功能是取消对<标识符>已有的宏定义。被取消了宏定义的标识符,可以对它重新进行定义。
      宏定义可以嵌套,已被定义的标识符可以用来定义新的标识符。例如:
      #define PI 3.14159265
      #define R 10
      #define AREA (PI*R*R)
      单的宏定义将一个标识符定义为一个字符串,源程序中的该标识符均以指定的字符串来代替。前面已经说过,预处理命令不同于一般C++语句。因此预处理命令后通常不加分号。这并不是说所有的预处理命令后都不能有分号出现。由于宏定义只是用宏名对一个字符串进行简单的替换,因此如果在宏定义命令后加了分号,将会连同分号一起进行置换。
      带参数的宏定义
      带参数的宏定义的一般形式如下:
      #define <宏名>(<参数表>) <宏体>
      其中, <宏名>是一个标识符,<参数表>中的参数可以是一个,也可以是多个,视具体情况而定,当有多个参数的时候,每个参数之间用逗号分隔。<宏体>是被替换用的字符串,宏体中的字符串是由参数表中的各个参数组成的表达式。例如:
      #define SUB(a,b) a-b
      如果在程序中出现如下语句:
      result=SUB(2, 3)
      则被替换为:
      result=2-3;
      如果程序中出现如下语句:
      result= SUB(x+1, y+2);
      则被替换为:
      result=x+1-y+2;
      在这样的宏替换过程中,其实只是将参数表中的参数代入到宏体的表达式中去,上述例子中,即是将表达式中的a和b分别用2和3代入。
      我们可以发现:带参的宏定义与函数类似。如果我们把宏定义时出现的参数视为形参,而在程序中引用宏定义时出现的参数视为实参。那么上例中的a和b就是形参,而2和3以及x+1和y+2都为实参。在宏替换时,就是用实参来替换<宏体>中的形参。
      在使用带参数的宏定义时需要注意的是:
      (1)带参数的宏定义的<宏体>应写在一行上,如果需要写在多行上时,在每行结束时,使用续行符 "\"结
      束,并在该符号后按下回车键,最后一行除外。
      (2)在书写带参数的宏定义时,<宏名>与左括号之间不能出现空格,否则空格右边的部分都作为宏体。
      例如:
      #define ADD (x,y) x+y
      将会把"(x,y)x+y"的一个整体作为被定义的字符串。
      (3)定义带参数的宏时,宏体中与参数名相同的字符串适当地加上圆括号是十分重要的,这样能够避免
      可能产生的错误。例如,对于宏定义:
      #define SQ(x) x*x
      当程序中出现下列语句:
      m=SQ(a+b);
      替换结果为:
      m=a+b*a+b;
      这可能不是我们期望的结果,如果需要下面的替换结果:
      m=(a+b)*(a+b);
      应将宏定义修改为:
      #define SQ(x) (x)*(x)
      对于带参的宏定义展开置换的方法是:在程序中如果有带实参的宏(如"SUB(2,3)"),则按"#define"命令行中指定的字符串从左到右进行置换。如果串中包含宏中的形参(如a、b),则将程序语句中相应的实参(可以是常量、变量或者表达式)代替形参,如果宏定义中的字符串中的字符不是参数字符(如a-b中的-号),则保留。这样就形成了置换的字符串。
评论读取中...

  • NCSA_BiGTooTh | 200
  • 帮助
  •   程序设计语言的预处理的概念:在编译之前进行的处理。
      C语言的预处理主要有三个方面的内容:
      1.宏定义;
      2.文件包含;
      3.条件编译。
      预处理命令以符号“#”开头。
    [编辑本段]
    一.宏定义
      1.不带参数的宏定义:
      宏定义又称为宏代换、宏替换,简称“宏”。
      格式:
      #define 标识符 字符串
      其中的标识符就是所谓的符号常量,也称为“宏名”。
      预处理(预编译)工作也叫做宏展开:将宏名替换为字符串。
      掌握"宏"概念的关键是“换”。一切以换为前提、做任何事情之前先要换,准确理解之前就要“换”。
      即在对相关命令或语句的含义和功能作具体分析之前就要换:
      例:
      #define PI 3.1415926
      把程序中出现的PI全部换成3.1415926
      说明:
      (1)宏名一般用大写
      (2)使用宏可提高程序的通用性和易读性,减少不一致性,减少输入错误和便于修改。例如:数组大小常用宏定义
      (3)预处理是在编译之前的处理,而编译工作的任务之一就是语法检查,预处理不做语法检查。
      (4)宏定义末尾不加分号;
      (5)宏定义写在函数的花括号外边,作用域为其后的程序,通常在文件的最开头。
      (6)可以用#undef命令终止宏定义的作用域
      (7)宏定义可以嵌套
      (8)字符串" "中永远不包含宏
      (9)宏定义不分配内存,变量定义分配内存。
      2.带参数的宏:
      除了一般的字符串替换,还要做参数代换
      格式:
      #define 宏名(参数表) 字符串
      例如:#define S(a,b) a*b
      area=S(3,2);第一步被换为area=a*b; ,第二步被换为area=3*2;
      类似于函数调用,有一个哑实结合的过程:
      (1)实参如果是表达式容易出问题
      #define S(r) r*r
      area=S(a+b);第一步换为area=r*r;,第二步被换为area=a+b*a+b;
      正确的宏定义是#define S(r) (r)*(r)
      (2)宏名和参数的括号间不能有空格
      (3)宏替换只作替换,不做计算,不做表达式求解
      (4)函数调用在编译后程序运行时进行,并且分配内存。宏替换在编译前进行,不分配内存
      (5)宏的哑实结合不存在类型,也没有类型转换。
      (6)函数只有一个返回值,利用宏则可以设法得到多个值
      (7)宏展开使源程序变长,函数调用不会
      (8)宏展开不占运行时间,只占编译时间,函数调用占运行时间(分配内存、保留现场、值传递、返回值)
    [编辑本段]
    二. 文件包含
      一个文件包含另一个文件的内容
      格式:
      #include "文件名"
      或
      #include <文件名>
      编译时以包含处理以后的文件为编译单位,被包含的文件是源文件的一部分。
      编译以后只得到一个目标文件.obj
      被包含的文件又被称为“标题文件”或“头部文件”、“头文件”,并且常用.h作扩展名。
      修改头文件后所有包含该文件的文件都要重新编译
      头文件的内容除了函数原型和宏定义外,还可以有结构体定义,全局变量定义:
      (1)一个#include命令指定一个头文件;
      (2)文件1包含文件2,文件2用到文件3,则文件3的包含命令#include应放在文件1的头部第一行;
      (3)包含可以嵌套;
      (4)<文件名>称为标准方式,系统到头文件目录查找文件,
      "文件名"则先在当前目录查找,而后到头文件目录查找;
      (5)被包含文件中的静态全局变量不用在包含文件中声明。
    [编辑本段]
    三. 条件编译
      有些语句行希望在条件满足时才编译。
      格式:(1)
      #ifdef 标识符
      程序段1
      #else
      程序段2
      #endif
      或
      #ifdef
      程序段1
      #endif
      当标识符已经定义时,程序段1才参加编译。
      格式:(2)
      #ifndef 标识符
      格式:(3)
      #if 表达式1
      程序段1
      #else
      程序段2
      #endif
      当表达式1成立时,编译程序段1,当不成立时,编译程序段2。
      使用条件编译可以使目标程序变小,运行时间变短。
      预编译使问题或算法的解决方案增多,有助于我们选择合适的解决方案。
      此外,还有布局控制:#progma,这也是我们应用预处理的一个重要方面,主要功能是为编译程序提供非常规的控制流信息。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值