预处理(二)#define宏替换

本文详细介绍了宏替换的两种形式,包括无参数和带参数的宏定义,强调了宏展开可能导致的副作用。同时,对比了#define与inline函数在展开时机、类型检查和二义性等方面的不同,并探讨了#define与const关键字在声明常量时的差异,const具有类型检查和防止意外修改等优点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一 宏替换:

*宏定义有两种形式:取决于被定义的宏名后面是否紧随一个左括号。


(1)#define 宏名 替换文本:最简单的宏替换。

*后续所有出现宏名的地方都将被替换为替换文本。

*#define中名字与变量名的命名方式相同,替换文本可以是任意字符串。

*通常#define指令占一行,替换文本是#define指令行尾部的所有剩余部分内容,也可以把一个较长的宏定义分成若干行,这时需要在行末尾加上反斜杠符\。


(2)#define 宏名(标识符列表) 标记序列:定义带参数的宏。优点:不会招致函数调用开销。

*标识符列表:逗号分隔的形式参数名称列表。

*左括号必须紧跟宏名之后,中间不能有空格。如果宏名和左括号之间被一个空格所分隔,则这个宏定义不接受任何参数,并且宏体从这个左括号开始。

*形式:#define max(a,b) ((a)>(b)?(a):(b))//注意不带参数类型。可用template inline函数替代。

*最好在宏定义中把每个参数都用括号括起来,同样整个表达式也应该用括号括起来。


(3)#define的问题:

*一个操作数如果在两处被用到,就会被求值两次。

如:int a=10,b=1;max(a++,b):使用宏返回值是11;使用普通函数返回10。

使用宏时宏展开后为:((a++)>(b)?(a++)):(b)).如果a>b,则a会被++两次。


二 #define与inline的区别:

1)内联在编绎时展开,宏在预编译时展开

2)编译内联函数可以嵌入到目标代码,宏只是简单文本替换。

3)内联会做类型,语法检查,而宏不具这样功能。

4)宏不是函数,inline函数是函数

5)宏定义小心处理宏参数(一般参数要括号起来),否则易出现二义性,而内联定义不会出现。


#define与const的区别:

(1) define是预处理指令,是简单的文字替换。const是关键字,用于变量声明的修饰。

(2) 编译器处理方式不同:
  define宏是在预处理阶段展开。
  const常量是编译运行阶段使用。
(3) 类型和安全检查不同:
  define宏没有类型,不做任何类型检查,仅仅是展开。
  const常量有具体的类型,在编译阶段会执行类型检查。
(4) 存储方式不同:
  define宏仅仅是展开,有多少地方使用,就展开多少次,不会分配内存。
  const常量会在内存中分配(可以是堆中也可以是栈中)。







评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值