c语言关于宏的使用十分频繁。但是宏的使用有利也有弊,与此同时,它还是一个特别容易搞错的地方。正是基于此,它常常成为一些面试会侧重考察的地方。

所谓宏就是 #define 机制包括的一个规定,即允许把参数替换到文本中。它的声明方式:#define name(参数列表) stuff

其中参数列表是一个由逗号分隔的符号列表,对应参数作用于stuff中,相当于宏替换函数;如果没有参数列表,那就是我们平常用得比较多的宏替换变量了。

使用特别要注意问题


1. 分号问题

2. 符号优先级问题

3. 作用域问题

4.使用带副作用的宏参数

宏定义处用了分号

#include <iostream>
using namespace std; #define MAX( l, r) l > r? l: r;
int main()
{
int a = , b=; if( a)
int max = MAX( a,b);// <=> max = a > b? a:b;; 出错!!
else
{
}
return ;
}

符号优先级问题

 #include <iostream>
using namespace std; #define MAX( l, r) l > r? l: r
int main()
{
int a = , b=; int max = MAX( a && , b);//a && 3 > b? a && 3: b 达不到预想结果
cout<<max;
return ;
}

改造:

 #define MAX( l, r) ( (l)> (r)? (l) :(r) )
int main()
{
int a = , b=; int max = MAX( a && , b);//(a && 3) > b? (a && 3) : b
cout<<max;
return ;
}

作用域问题

 #include <iostream>
using namespace std;
//=优先级很低,不用优先级问题
#define SWAP(l, r) {\
int tmp = l;\
l = r; \
r = tmp;}
int main()
{
int a = , b=;
if(a)
SWAP(a, b);
//<=> {.... } ; 此处只允许出现一条语句,但这里产生2条
else
{
}
cout<<a <<" "<<b<<endl;
return ;
}

解决:用do{... }while(0)  解决

 //=优先级很低,不用优先级问题
#define SWAP(l, r) do{\
int tmp = l;\
l = r; \
r = tmp;}while()
int main()
{
int a = , b=;
if(a)
SWAP(a, b); //<=> do{.... }while( 0) ;
else
{
}
cout<<a <<" "<<b<<endl;
return ;
}

带副作用的宏参数

当宏参数在宏定义中出现次数超过一次时,如果这个参数具有副作用,那么当使用这个宏时就出现危险,导致不可预料的结果。副作用就是在表达式求值时出现永久性的效果。比如x++,它可以增加x的值。当下一次执行该表达式时,他将产生一个全新的结果。还是以前面的的MAX函数为例

 #include <iostream>
using namespace std; #define MAX(l, r) ( (l)>(r)? (l): (r))
int main()
{
int a = , b=;
int max = MAX(a++, b++); //<=> (a++)>( b++)? ( a++):( b++);
//结果: 3 2 4
cout<<max<<" "<<a<<" "<<b<<endl;
return ;
}

结果让较小的值a增加了一次,但确让较大的值b增加了2次。这种带副作用宏参数会修改变量的值,使用需格外注意。

宏的优缺点


优点

1. 快! 由于是预处理时期直接宏替换,不用像函数那样来回调用返回,增加额外开销。

2. 由于可以替换变量,修改变量仅需在宏定义处修改,增加了程序的可维护性。

缺点:

1. 没有类型安全的检查。宏和类型是无关的,只要对参数操作合法,它可以使用任何参数类型。

2. 极易出错。  宏参数求值是要依赖于周围表达式的上下文环境。没有合理地加括号,得不到期望的结果; 同时参数每次用于宏定义时,它们都会重新求值,由于这样多次求值,所以让具有副作用的参数可能产生不可预料的结果。

3. 不可调试。由于预处理阶段,直接进行了宏替换,对替换掉的代码无法进行调试检查。

4. 替换插入代码,导致程序代码长度大大加长。

C语——宏小结的更多相关文章

  1. 如何编译和调试Python内核源码?

    目录 写在前面 获取源代码 源代码的组织 windows下编译CPython 调试CPython 小结 参考 博客:blog.shinelee.me | 博客园 | CSDN 写在前面 如果对Pyth ...

  2. 【手摸手,带你搭建前后端分离商城系统】01 搭建基本代码框架、生成一个基本API

    [手摸手,带你搭建前后端分离商城系统]01 搭建基本代码框架.生成一个基本API 通过本教程的学习,将带你从零搭建一个商城系统. 当然,这个商城涵盖了很多流行的知识点和技术核心 我可以学习到什么? S ...

  3. C++ macro(宏)使用小结

    谈起C++中的宏,我们第一个想到的应该就是“#define”,它的基本语法长得像这样: #define macroname(para1, para2, para3, ... ,paran) macro ...

  4. IOS开发,知识点小结,ios开发中经常使用的宏定义总结

    IOS开发,从应用跳转到用浏览器打开网页: [[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"http:// ...

  5. C语言中do...while(0)用法小结

    在linux内核代码中,经常看到do...while(0)的宏,do...while(0)有很多作用,下面举出几个: 本文地址:http://www.cnblogs.com/archimedes/p/ ...

  6. C语言中,宏和全局变量的区别是什么?

    全局变量 是可以在程序中任何地方使用 而且是可以修改的 宏定义也可以在任何地方使用 但是不能在之后修改 数据类型没有限制的 宏的例子:#define 宏名 宏体 #define PI 3.141592 ...

  7. [转载]C宏定义的小结

    FROM:http://blog.csdn.net/sunboy_2050/article/details/6103530 实现代码实例 程序代码: #include <stdio.h> ...

  8. 【原创】mdk5宏定义的使用小结

    前几天在网上申请了一块芯片为stm32f103cbt6的小板子. 在用keil编程的过程中发现一些小问题,总结如下: 使用mdk5开发,与之前的库函数的引用方式已经改变.不需要在选项的c/c++标签页 ...

  9. 小结:Swift、OC语言中多target在代码中如何区分

    一.对swift工程 经实践,网上的方法都无法成功,后来思考DEBUG宏定义方式,经实测有效,方式如下: 注意:不能把swift flags 小三角折叠后双击设置-DTarget4AppStore, ...

随机推荐

  1. centos常用网络管理命令

    网卡配置命令:ifconfig (ip addr , ip link) ifconfig:显示所有活动状态的相关信息    ifconfig Interface:仅显示指定接口的相关信息    ifc ...

  2. lamp环境搭建之配置apache与fpm方式的php

    配置apache-2.4.9与fpm方式的php-5.4.26 一.apache.MySQL的安装参考<编译安装lamp环境> http://blog.csdn.net/reblue520 ...

  3. php ajax返回无故刷新页面

    1 前言 一个php页面,里面两个$.POST请求,一个会刷新页面,一个不会,然后就拉出来研究一下了,仅作为记录使用. 2 代码 HTML代码: <input value="查找&qu ...

  4. node 相关网站

    包管理网站:https://www.npmjs.com/

  5. 测试开发之前端——No4.HTML5中的事件属性

    HTML5的事件属性. 属性 值 描述 onafterprint script 在打印文档之后运行脚本 onbeforeprint script 在文档打印之前运行脚本 onbeforeonload ...

  6. 11:django 模板 内建标签

    django 内建标签 autoescape 控制当前自动转义的行为,有on和off两个选项 {% autoescape on %} {{ body }} {% endautoescape %} bl ...

  7. RzPageControl Tab拖拽 移动

  8. django----数据库表设计

    设计表时注意的几点: 1. nid = models.AutoField(primary_key=True)        #如果不指定django会默认加上id的 nid = models.BigA ...

  9. 【BZOJ4919】[Lydsy六月月赛]大根堆

    题解: 我觉得数据结构写成结构体还是有必要的 因为不然一道题里出现了两个相同的数据结构由于名字很像很容易出错 另外初始化用segmenttree(){ } 首先裸的dp很好想 f[i][j]表示在i点 ...

  10. java:冒泡排序、选择排序、插入排序实现

    整数排序 给一组整数,按照升序排序,使用选择排序,冒泡排序,插入排序或者任何 O(n2) 的排序算法. 样例 样例 1: 输入: [3, 2, 1, 4, 5] 输出: [1, 2, 3, 4, 5] ...