宏的副作用和do{}while(0)的说明

2024-05-07 11:18
文章标签 说明 副作用

本文主要是介绍宏的副作用和do{}while(0)的说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文如下:

    问题:写一个“标准”宏MIN,这个宏输入两个参数并返回较小的一个,另外,当你写下面的代码时会发生什么事?

           least=MIN(*p++,b);

解答:

    #define MIN(A,B) ((A)<=(B)?(A):(B))      //注意加括号的方法

    当使用MIN(*p++,b)时会产生宏的副作用;


剖析:

    这个面试题主要考察面试者对宏定义的使用,宏定义可以实现类似于函数的功能,但是它终归不是函数,而宏定义中括弧中的"参数"也不是真的参数,在宏展开时对"参数"进行的是一对一的替换;


程序员对宏定义的使用要非常小心,特别要注意两个问题:

(1)、谨慎地将宏定义中的"参数"和整个宏用括号扩起来;

    所以,严格地讲,下述解答都是错误的:

    #define MIN(A,B)  (A)<=(B)?(A):(B)

    #define MIN(A,B) (A<=B?A:B)

(2)、防止宏的副作用:

     #define MIN(A,B) ((A)<=(B)?(A):(B))对于MIN(*p++,b)的作用结果为((*p++)<=(b)?(*p++):(b))   *p会做两次自加操作;

    除此之外,另一个错误的解答是#define MIN(A,B) ((A)<=(B)?(A):(B));这个解答在宏定义的后面加了一个逗号,显示编写者对宏的概念模糊不清,也是错误的;


#include <stdio.h> 
#include<stdlib.h>
#define min_i(x, y) ((x) < (y) ? (x) : (y))
//(1)

#define min_t(type, x, y) do{ type __x = x; \
type __y = y;\
__x < __y ? __x : __y;}while(0);
#define min(x, y) { const int _x = (x); const int _y = (y); (void) (&_x == &_y);\
 printf("%d\n",_x < _y ? _x : _y);   };
int main()
{   
int a = 10;  int b = 20;   
printf("min_i(a++, b++) = %d\n", min_i(a++, b++)); // 11 
        printf("a = %d\n",a); // 12  
printf("b = %d\n",b); // 21    
int c = 10;  char d = 'a';   
        min_t(int, c++, d++); // 10  
printf("a = %d\n", a); // 11  
printf("b = %d\n", b); // 21    
a = 10;  b = 20;   
min(a++, b++); // 10  
printf("a = %d\n", a); // 11  
printf("b = %d\n", b); // 21 
system("PAUSE");
return 0;
}

(1)、这种定义计算x和y分别两次(其实是x和y中的小者被计算两次),当参数有副作用时,将产生不正确的结果;

(2)、后面两种定义只计算参数一次,避免了可能的错误:语句表达式通常用于宏定义;

do{}while(0)的作用:http://www.cnblogs.com/flying_bat/archive/2008/01/18/1044693.html

来源:

http://wenku.baidu.com/link?url=-vpjW5-L5AhYC60PBUteUgoVTtZsLgnGDVeh9pAVLu4PQK4Y4rucfzyVzwDmDHRTzXL3uRqz3RoK-WoYgCRmjUEmPkKkNkk-EjKGaiQ3hTy(自己做了一些修改)


这篇关于宏的副作用和do{}while(0)的说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/967194

相关文章

Redis中哨兵机制和集群的区别及说明

《Redis中哨兵机制和集群的区别及说明》Redis哨兵通过主从复制实现高可用,适用于中小规模数据;集群采用分布式分片,支持动态扩展,适合大规模数据,哨兵管理简单但扩展性弱,集群性能更强但架构复杂,根... 目录一、架构设计与节点角色1. 哨兵机制(Sentinel)2. 集群(Cluster)二、数据分片

Springboot项目构建时各种依赖详细介绍与依赖关系说明详解

《Springboot项目构建时各种依赖详细介绍与依赖关系说明详解》SpringBoot通过spring-boot-dependencies统一依赖版本管理,spring-boot-starter-w... 目录一、spring-boot-dependencies1.简介2. 内容概览3.核心内容结构4.

redis和redission分布式锁原理及区别说明

《redis和redission分布式锁原理及区别说明》文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,... 目录Redis和redission分布式锁原理及区别1、有的同伴想到了synchronized关键字

MySQL 临时表创建与使用详细说明

《MySQL临时表创建与使用详细说明》MySQL临时表是存储在内存或磁盘的临时数据表,会话结束时自动销毁,适合存储中间计算结果或临时数据集,其名称以#开头(如#TempTable),本文给大家介绍M... 目录mysql 临时表详细说明1.定义2.核心特性3.创建与使用4.典型应用场景5.生命周期管理6.注

Java中数组与栈和堆之间的关系说明

《Java中数组与栈和堆之间的关系说明》文章讲解了Java数组的初始化方式、内存存储机制、引用传递特性及遍历、排序、拷贝技巧,强调引用数据类型方法调用时形参可能修改实参,但需注意引用指向单一对象的特性... 目录Java中数组与栈和堆的关系遍历数组接下来是一些编程小技巧总结Java中数组与栈和堆的关系关于

mybatis-plus QueryWrapper中or,and的使用及说明

《mybatis-plusQueryWrapper中or,and的使用及说明》使用MyBatisPlusQueryWrapper时,因同时添加角色权限固定条件和多字段模糊查询导致数据异常展示,排查发... 目录QueryWrapper中or,and使用列表中还要同时模糊查询多个字段经过排查这就导致只要whe

SpringBoot改造MCP服务器的详细说明(StreamableHTTP 类型)

《SpringBoot改造MCP服务器的详细说明(StreamableHTTP类型)》本文介绍了SpringBoot如何实现MCPStreamableHTTP服务器,并且使用CherryStudio... 目录SpringBoot改造MCP服务器(StreamableHTTP)1 项目说明2 使用说明2.1

JAVA覆盖和重写的区别及说明

《JAVA覆盖和重写的区别及说明》非静态方法的覆盖即重写,具有多态性;静态方法无法被覆盖,但可被重写(仅通过类名调用),二者区别在于绑定时机与引用类型关联性... 目录Java覆盖和重写的区别经常听到两种话认真读完上面两份代码JAVA覆盖和重写的区别经常听到两种话1.覆盖=重写。2.静态方法可andro

zookeeper端口说明及介绍

《zookeeper端口说明及介绍》:本文主要介绍zookeeper端口说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、zookeeper有三个端口(可以修改)aVNMqvZ二、3个端口的作用三、部署时注意总China编程结一、zookeeper有三个端口(可以

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化