c 宏定义用法#define
转自:https://blog.csdn.net/boring_wednesday/article/details/78756696
宏定义
语法
#define name Stuff
#define PI 3.14
//定义一个M,值为3.14
#define DO_FOREVER for(;;)
//定义一个死循环
#define REG register
//定义REG来作为register的别名
#define CASE break;case
//在switch中用CASE来补上break;
#define DEBUG_PRINT printf("file:%s\tline:%d\tdate:%s\ttime:%s\n",\
__FILE__, __LINE__, __DATE__, __TIME__);
//测试预定义符号
1
2
3
4
5
6
7
8
9
10
11
1、宏的作用范围
先看下面代码:
欢迎访问jo-qzy的博客
让我们查看上图中代码经过预处理后的样子
欢迎访问jo-qzy的博客
可以发现宏只对宏定义后的行数起作用,且与定义在哪里无关,即使函数不被调用,也可以使用宏
2、宏替换的原则
在程序中扩展#define定义符号和宏时,需要涉及几个步骤。
1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先被替换。
2. 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。
3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。
注意:
1、宏函数不能出现递归
2、宏定义的符号,即name部分不会在预处理替换的时候被搜索
3、宏定义#define后不需要加;
例如:
#define M 100;
这里在100后面加上了;
在句子中有时候就会出现问题
观察下面代码片段:
#define M 100;
if (condition)
m = M;
else
max = 0;
1
2
3
4
5
在该选择语句中会出现语法错误,没有与else与之匹配的if语句
4、宏函数申明
宏函数申明格式:
#define name ( parament-list ) stuff
parament-list为参数表,可以包含多个参数,他们会在stuff出现
例如:
#define SQARE( X ) X*X
//定义一个计算乘方的宏函数
1
2
但是这么定义是会出现预料之外的错误的,观察下面代码片段
#define SQUARE(X) X*X
int main()
{
int a = 5;
printf("%d\n", SQUARE(a+1));
return 0;
}
1
2
3
4
5
6
7
8
本来结果应该为6^2 = 36
欢迎访问jo-qzy的博客
但是实际计算机输出的结果为11
我们用gcc观察预处理后的代码片段是怎么样的
使用命令$ gcc -E test.c -o test.i来查看预处理后的代码
欢迎访问jo-qzy的博客
可以看到这里原式被替换成 a + 1 * a + 1 = 11,而这个结果显然不是我们期望的
这里我们提出解决方案,将 X 用括号括起来(X),这样就避免上述代码因符号优先级带来的错误
至此,上面代码解决了,来看下面的宏函数定义:
#define DOUBLE(X) (X) + (X)
int main()
{
int a = 5;
printf("%d\n", 10 * DOUBLE(a));
return 0;
}
1
2
3
4
5
6
7
8
期望结果为100,而看程序执行结果:
欢迎访问jo-qzy的博客
程序再次出现预料之外的结果,原因是原式被替换成了
10 * 5 + 5 = 55
解决方法:
在宏函数定义时,对Stuff中的参数以及结果均用括号来避免因符号运算优先级带来的问题
5、宏中的#和##
#的用法
首先要理解一个原则,即邻近字符串连接原则
在C语言中
printf(“hello”” world!”“\n”);
这句话是合法的
打印结果为:
hello world!
按照上述原则,我们可以写出下列代码:
#define PRINT(FORMAT, VALUE) printf("the value of " #VALUE " is "FORMAT"\n", VALUE)
int main()
{
int i = 0;
PRINT("%d", i + 3);
return 0;
}
1
2
3
4
5
6
7
8
看下gcc编译后的代码,程序正常运行:
欢迎访问jo-qzy的博客
我们再通过gcc生成test.i文件看下预处理是怎么样的:
欢迎访问jo-qzy的博客
所以我们可以看出来#的作用:
将一个宏参数变成一个对应的字符串
在上述例子中:
#VALUE被替换成了“i + 3”
别忘了被替换的时候i + 3两边加上了双引号
##的用法
看下面的代码片段
#define ADD_TO_AN(num, value) a##num += value
int main()
{
int a1 = 0;
int a2 = 0;
ADD_TO_AN(1, 5);
ADD_TO_AN(2, 6);
return 0;
}
1
2
3
4
5
6
7
8
9
10
解释一下上面这段代码
假设有一个变量叫a1
此时代码片段为ADD_TO_AN(1, 5)
即替换为a1 += 5,给a1变量增加5
同理ADD_TO_AN(2, 6)则替换成a2 += 6
##的作用
将##两边的字符连在一起作为一个标识符
前提连接后的标识符必须合法,否则编译出现标识符未定义
6、宏和函数
宏通常被应用于执行简单的运算
和函数相比,宏有他的优点
宏的优点:
1. 用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多。所以宏比函数在程序的规模和速度方面更胜一筹
2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使用。反之这个宏怎可以适用于整形、长整型、浮点型等可以用于>来比较的类型。宏的参数与类型无关的
3. 宏参数可以使用变量类型,而函数不可以,例如:
#define MALLOC(num, type) (type *)malloc(num * sizeof(type))
MALLOC(10, int);//类型作为参数
//预处理器替换之后:
(int *)malloc(10 * sizeof(int));
1
2
3
4
5
但是,宏参数与类型无关是一把双刃剑,和函数比较也有他的缺点
宏的缺点:
1. 每次使用宏的时候,一份宏定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序的长度
2. 宏是没法调试的
3. 宏由于类型无关,也就不够严谨
4. 宏可能会带来运算符优先级的问题,导致程序容易出现问题
我的建议:
当有一部分功能既可以用函数实现也可以用宏实现,且在调用函数的过程与宏使用过程所消耗的资源相当时,优先考虑代码的严谨性,使用函数
7、宏参数的副作用
当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果。
#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
x = 5;
y = 8;
z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);
//结果为x=6 y=10 z=9
1
2
3
4
5
6
7
可以看到上面代码中y++带来了副作用,所以在使用宏的时候,避免使用++运算和--运算
8、宏的命名
宏和函数的使用很类似,我们平时的使用习惯是:
1. 宏名全部大写
2. 函数名不全大写
---------------------
作者:jo-qzy
来源:CSDN
原文:https://blog.csdn.net/boring_wednesday/article/details/78756696
版权声明:本文为博主原创文章,转载请附上博文链接!
相关文章:

Linux学习笔记—— 权限及权限管理
权限及权限管理权限管理:r:w:x:三类用户:u:属主g:属组o:其他用户chown:改变文件属主(只有管理员可以使用此命令)# chown USERNAME file,...-R&…
【ACM】签到题
#include <stdio.h> int main () {int T,a,b,c,x,ji,ya,e;scanf("%d",&T);while(T--){scanf("%d%d%d%d",&a,&b,&c,&x);ya(a*x)/(c-b);e(a*b*x)/(a*c-a*b);jiex;printf("%d %d %d\n",ji,ya,e);}return 0; }
图解eclipse+myeclipse完全绿色版制作过程
现在在Java开发中,使用的开发工具大部分都是Eclipse,并且和Eclipse关系紧密的要数MyEclipse了,但是 MyEclipse是一个EXE可执行程序,对于没有安装Eclipse与MyEclilpse的电脑来说,首先得先解压Eclipse,然后再…

linux proc/xx/maps文件分析
转载:https://blog.csdn.net/lijzheng/article/details/23618365 Proc/pid/maps显示进程映射了的内存区域和访问权限。对应内核中的操作集为proc_pid_maps_op,具体的导出函数为show_map。内核中进程的一段地址空间用一个vm_area_struct结构体表示&#…
【ACM】熊孩子的乐趣
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1000 【问题描述】 Alice跟Bob是学校里出了名的两个熊孩子,会在任何事情上争个高低,彼此都不服输。幼儿园的老师每次分糖果的时候看到这两个熊孩子也很头疼,两个人都想占便宜…

mysql insertOrUpdate 方法
为什么80%的码农都做不了架构师?>>> 自己对这个方法又有点小心得 分享下 https://my.oschina.net/hccake/blog/777225 mysql "ON DUPLICATE KEY UPDATE" 语法 如果在INSERT语句末尾指定了ON DUPLICATE KEY UPDATE,并且插入行后会导…

软件破解工具整理收集
1.调试工具softice 2.调试工具Trw2000 3.反汇编工具Wdasm8.93 4.Hiew 5.Visual Basic程序调试工具Smartcheck 6.十六进制编辑器(如:Ultraedit、WinHex、Hex Workshop 等) 7.注册表监视工具RegShot、regmon或RegSnap 8.侦测文件类型工具…

Linux 下 UltraEdit 版本: 16.1.0.18 破解 30 天试用限制
rm -rfd ~/.idm/uex rm -rf ~/.idm/*.spl rm -rf /tmp/*.spl

【ACM】杭电OJ 2149
Public Sale 【问题描述】 虽然不想,但是现实总归是现实,Lele始终没有逃过退学的命运,因为他没有拿到奖学金。现在等待他的,就是像FarmJohn一样的农田生涯。 要种田得有田才行,Lele听说街上正在举行一场别开生面的拍…

PV UV IP
UV (网站独立访客) 编辑UV是unique visitor的简写,是指通过互联网访问、浏览这个网页的自然人。独立IP:是指独立用户/独立访客。指访问某个站点或点击某条新闻的不同IP地址的人数,在同一天的00:00-24:00内,…
VBA中级班课时3小结
本课内容:工作簿和工作表对象 主讲:rover18 学习时间:2010年11月 本节课将学习工作簿对象Workbooks、Workbook与工作表对象Worksheets、Worksheet。在我们了解了VBA的四大要素——对象、属性、方法和事件后,会发现VBA的程序是对对…

解决firefox ubuntu无法打开页面的问题
firefox备份用户配置信息 https://support.mozilla.org/zh-CN/kb/%E5%A4%87%E4%BB%BD%E4%BD%A0%E7%9A%84%E4%BF%A1%E6%81%AF 把xxxxxxxx.default 覆盖掉xxxxxxxx.default-release里面的内容

js构造函数式编程
1.函数式编程 //创建和初始化地图函数:function initMap(){createMap();//创建地图setMapEvent();//设置地图事件addMapControl();//向地图添加控件}//创建地图函数:function createMap(){var map new BMap.Map("dituContent");//在百度地图容…
【ACM】绝地求生
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1009 【问题描述】 zbt最近喜欢上了《绝地求生》(pubg)游戏,pubg这个游戏有一种跑毒机制,每次会产生一个圆形的安全区,玩家需要从他的当前位置在一定时间内进入安…
【oracle】dblink创建
目的:oracle中跨数据库查询 两台数据库服务器db_A(本地)和db_B(远程192.168.1.100),db_A下用户user_a 需要访问到db_B下user_b的数据解决:查询得知使用dblink(即database link 数据库链)实现过程:1、确定用户user_a有没有创建 db…

ASan(Linux),gcc4.8以上版本自带的内存检查工具
转自:http://shafeng.github.io/2017/05/10/asan/ 最近线上的程序总是莫名其妙崩溃,因为我们的项目使用了分布负载的机制,对于玩家的影响其实很小,但是我肯定是忍不了的…程序崩溃的core文件里面完全找不到问题所在,初步分析应该是野指针导致,仔细分析程序之后并没有…

详解使用DockerHub官方的mysql镜像生成容器
为什么80%的码农都做不了架构师?>>> 写在前面:看到网上关于利用DockerHub官方的mysql镜像生成容器此类的文档比较少,故结合自身实践分享给大家,还望多多指教。 我的需求:利用docker 镜像快速建立一个mysql…
【ACM】奇怪的回文数
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1008 【问题描述】 “回文”是指正读反读都能读通的句子,它是古今中外都有的一种修辞方式和文字游戏,如“我为人人,人人为我”等。 在数学中也有这样一类数字有这样的特征…
java I/O之装饰者模式
装饰者: Decorator模式(别名Wrapper):动态将职责附加到对象上,若要扩展功能,装饰者提供了比继承更具弹性的代替方案。 装饰者模式意图: 动态的给一个对象添加额外的职责。Decorator比生产子类灵…

ubuntu下wireshark添加root权限
wireshark要监控eth0,但是必须要root权限才行。但是,直接用root运行程序是相当危险,也是非常不方便的。 解决方法如下: 1.添加wireshark用户组sudo groupadd wireshark 2.将dumpcap更改为wireshark用户组sudo chgrp wireshark /…

Oracle导出空表解决办法
在oracle 11g 中,发现传统的exp不能导出空的表 oracle 11g 新增了一个参数:deferred_segment_creation,含义是段延迟创建,默认是true。具体是什么意思呢? 如果这个参数设置为true,你新建了一个表T1…

【ACM】图像分类
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1003 【问题描述】 现在, 我们需要你来解决一项图像分类任务。 首先我们需要介绍下简单图像的数据存储形式,你可以粗略的认为图像在数字意义就是一个二维矩阵(我们这里不考虑…
【译】如何精确判断最终用户响应时间过长的原因?
译者:原始文章有点性能测试工具软文的感觉,毕竟文章来源于某工具官方博客。高手请略过。 对于我这种新手,此文还是给我带来一些惊喜,从上到下地,从表象到根源地,定位他们遇到性能问题-响应时间过长-的根本原…

javascript中重要概念-闭包-深入理解
在上次的分享中javascript--函数参数与闭包--详解,对闭包的解释不够深入。本人经过一段时间的学习,对闭包的概念又有了新的理解。于是便把学习的过程整理成文章,一是为了加深自己闭包的理解,二是给读者提供学习的途径,…

ssl握手过程和ca证书验证
转载:https://www.cnblogs.com/cposture/p/9029014.html SSL 认证 可以将 SSL 服务器与客户端之间的通信配置为使用单向或双向 SSL 认证。 单向 SSL 认证一般是客户端利用服务器传过来的信息验证服务器的合法性,服务器的合法性包括:证书是…
【ACM】练武奇才
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1005 【问题描述】 很久很久以前,constbh大神还在上着小学。一天,在放学的路上,他被一位乞丐叫住,这位乞丐对constbh说,我看你骨骼惊奇,…

Bat命令学习
参考资料:http://www.cnblogs.com/SunShineYPH/archive/2011/12/13/2285570.html

记一次CentOS7内核kernel的删除重装
人生在于折腾,学习Linux更要多多折腾。在一次折腾中吸取教训,更易于记忆。今天我们来折腾Linux的内核:删除系统内核后,通过光盘进行kernel的重安装。友情提示:请在虚拟机环境进行,折腾前务必做好系统快照。…

tcpdump抓包并保存到远程服务器
有的时候,运行tcpdump抓包进程的主机A可能没有足够的硬盘空间。例如我们使用树霉派搭建了一个热点,然后我们想在树霉派上抓包,因为树霉派的存储很小,所以很容易在短时间内将存储空间使用完。 为了解决该问题,我们可以…
【ACM】家喻户晓的中药店(待更)
题目链接:http://acm.nuc.edu.cn/OJ/contest/show/43/1007 【问题描述】 long_xiao和const_hhh是一对恩爱的夫妻。 他们在京城经营着一家中药店,夫妻二人医术精湛、古道热肠,虽然年过花甲,身体依然硬朗。更重要的是ÿ…