C/C++里危险的宏(Macro)

#define SQUARE(x) x*x

上述宏定义SQUARE(x)用于求“参数”x的平方,这个宏很容易被使用者误认为是函数。宏是由预处理器处理的,它有着函数的形式却没有函数调用的代价。我们不建议初学者使用宏,因为使用宏的收益远不足以抵消其带给初学者的风险。

本文引用自作者编写的下述图书; 本文允许以个人学习、教学等目的引用、讲授或转载,但需要注明原作者"海洋饼干叔
叔";本文不允许以纸质及电子出版为目的进行抄摘或改编。
1.《Python编程基础及应用》,陈波,刘慧君,高等教育出版社。免费授课视频 Python编程基础及应用
2.《Python编程基础及应用实验教程》, 陈波,熊心志,张全和,刘慧君,赵恒军,高等教育出版社Python编程基础及应用实验教程
3. 《简明C及C++语言教程》,陈波,待出版书稿。免费授课视频

//Project - Macro
#include <iostream>
using namespace std;

#define SQUARE(x)  x*x

int main() {
    cout << SQUARE(3) << endl;      //3的平方,预期输出为9
    cout << SQUARE(3+2) << endl;    //3+2的平方,预期输出为5的平方,即25
    return 0;
}

对于SQUARE(x)宏,从使用者的角度看来,特别像函数。按照函数的理解,SQUARE(3+2)即为SQUARE(5),预期结果应为25。但程序的实际执行结果却出人意料:

9
11

执行结果不符预期的原因与预处理器的工作原理有关,宏定义SQUARE(x)是在预处理阶段由预处理器展开的,而展开的方式就是简单的文本替换。对于预处理器而言,SQUARE(3+2)中的x即为“3+2”,按定义展开后,即为:

3+2*3+2

显然,按照先乘除、后加减的计算顺序,上述展开式的结果只能是3+6+2=11。

我们使用下述命令对上述源代码文件main.cpp进行了预处理,得到输出文件main.i:

gcc -E main.cpp -o main.i

在main.i中,我们可以看到预处理器对SQAURE(x)进行宏展开后的结果:

int main() {
    cout << 3*3 << endl;
    cout << 3+2*3+2 << endl;
    return 0;
}

如上所示,SQUARE(3)经预处理后变成了33;SQUARE(3+2)经预处理后变成了3+2*3+2。

象这类因为宏的不恰当定义所造成了软件缺陷,很难发现。我们再次建次初学者不要使用宏,如果期望避免不必要的函数调用代价,可将函数内联。

为了帮助更多的年轻朋友们学好编程,作者在B站上开了两门免费的网课,一门零基础讲Python,一门零基础C和C++一起学,拿走不谢!

简洁的C及C++
由编程界擅长教书,教书界特能编程的海洋饼干叔叔打造
Python编程基础及应用
由编程界擅长教书,教书界特能编程的海洋饼干叔叔打造

如果你觉得纸质书看起来更顺手,目前Python有两本,C和C++在出版过程中。

Python编程基础及应用

Python编程基础及应用实验教程
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值