C语——宏小结
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语——宏小结的更多相关文章
- 如何编译和调试Python内核源码?
目录 写在前面 获取源代码 源代码的组织 windows下编译CPython 调试CPython 小结 参考 博客:blog.shinelee.me | 博客园 | CSDN 写在前面 如果对Pyth ...
- 【手摸手,带你搭建前后端分离商城系统】01 搭建基本代码框架、生成一个基本API
[手摸手,带你搭建前后端分离商城系统]01 搭建基本代码框架.生成一个基本API 通过本教程的学习,将带你从零搭建一个商城系统. 当然,这个商城涵盖了很多流行的知识点和技术核心 我可以学习到什么? S ...
- C++ macro(宏)使用小结
谈起C++中的宏,我们第一个想到的应该就是“#define”,它的基本语法长得像这样: #define macroname(para1, para2, para3, ... ,paran) macro ...
- IOS开发,知识点小结,ios开发中经常使用的宏定义总结
IOS开发,从应用跳转到用浏览器打开网页: [[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"http:// ...
- C语言中do...while(0)用法小结
在linux内核代码中,经常看到do...while(0)的宏,do...while(0)有很多作用,下面举出几个: 本文地址:http://www.cnblogs.com/archimedes/p/ ...
- C语言中,宏和全局变量的区别是什么?
全局变量 是可以在程序中任何地方使用 而且是可以修改的 宏定义也可以在任何地方使用 但是不能在之后修改 数据类型没有限制的 宏的例子:#define 宏名 宏体 #define PI 3.141592 ...
- [转载]C宏定义的小结
FROM:http://blog.csdn.net/sunboy_2050/article/details/6103530 实现代码实例 程序代码: #include <stdio.h> ...
- 【原创】mdk5宏定义的使用小结
前几天在网上申请了一块芯片为stm32f103cbt6的小板子. 在用keil编程的过程中发现一些小问题,总结如下: 使用mdk5开发,与之前的库函数的引用方式已经改变.不需要在选项的c/c++标签页 ...
- 小结:Swift、OC语言中多target在代码中如何区分
一.对swift工程 经实践,网上的方法都无法成功,后来思考DEBUG宏定义方式,经实测有效,方式如下: 注意:不能把swift flags 小三角折叠后双击设置-DTarget4AppStore, ...
随机推荐
- NoNodeAvailableException[None of the configured nodes are available: [{#transport#-1}{HBmUtjMOQP2pgLFFwqa_Og}{172.16.0.163}{172.16.0.163:9300}] ]
1.找到elasticsearch的安装目录,在config目录找到elasticsearch.yml,查看cluster.name的赋值 2.在SpringBoot的yml文件中,不仅要配置clus ...
- 读SRE Google运维解密有感(三)
前言 这是读“SRE Google运维解密”有感第三篇,之前的文章可访问www.addops.cn来查看.我们今天来聊聊“on call”也就是运维值班制度, 本人到目前为止也还在参与一线运维的值班, ...
- UNIX网络编程读书笔记:I/O模型(阻塞、非阻塞、I/O复用、信号驱动、异步)
I/O模型 UNIX下可用的5种I/O模型: (1)阻塞I/O (2)非阻塞I/O (3)I/O复用(select和poll) (4)信号驱动I/O(SIGIO) (5)异步I/O 对于一个套接口上的 ...
- Expm 4_2 有向无环图中的最短路径问题
[问题描述] 建立一个从源点S到终点E的有向无环图,设计一个动态规划算法求出从S到E的最短路径值,并输出相应的最短路径. 解: package org.xiu68.exp.exp4; import j ...
- vw 解决方案
vw 解决方案 1. 安装并配置PostCss插件 复制代码代码如下: npm i postcss-aspect-ratio-mini postcss-px-to-viewport postcss-w ...
- 【ES】学习3-请求体查询
1.空查询 GET /index_2014*/type1,type2/_search {} GET /_search { , } 2.查询表达式 DSL只需将查询语句传递给 query 参数 GET ...
- poj1742 多维背包
普通的多维背包做不了,需要优化一下 但是没有学优化..别的方法也是可以做的 省去一个 表示阶段的 i 维度,dp[j]表示面值为j的钱是否被凑出来了,used[j]表示第i种硬币在凑面值为j的时候被用 ...
- poj3349 散列表(hash)
就是散列表的应用,把每片哈希值相同的雪花排到一条链上去即可,每片雪花x的哈希值 hash(x)=sum(x的六角)+mul(x的六角),会爆int #include<iostream> # ...
- hdu4533 线段树维护分段函数
更新:x1,y1,x2,y2不用long long 会wa.. #include<iostream> #include<cstring> #include<cstdio& ...
- centOS下单点部署k8s
Kubernetes 是Google的一种基于容器的开源服务编排解决方案,在我们进行Kubernetes的学习前,为了对Kubernetes的工作有一个大概的认识, 我们需要先安装一个单节点的实例服务 ...