C语言系列-带有副作用的宏参数#和##命名约定宏替换的规则

2024-02-18 19:36

本文主要是介绍C语言系列-带有副作用的宏参数#和##命名约定宏替换的规则,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 🌈个人主页: 会编辑的果子君

💫个人格言:“成为自己未来的主人~”  

目录

带有副作用的宏参数

宏替换的规则

宏函数的对比

 #和##

#运算符

##运算符

命名约定

#undef


带有副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,不可控的后果,副作用就是表达式求值的时候出现的永久性效果

例如:

x + 1;//没有副作用
x++;  //有副作用

MAX宏可以证明具有副作用的参数所引起的问题:

#define MAX(a,b) ((a)>(b)?(a):(b))
#include<stdio.h>
int main()
{int x = 5;int y = 8;int z = MAX(x++, y++);printf("%d,%d,%d", x, y, z);return 0;
}

输出结果为 6 10 9

宏替换的规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤

  1. 在调用宏的时候,首先对参数进行检查,看看是否包含任何由#define定义的符号,如果有,他们首先被定义。
  2. 替换文本随后被插入到程序原来文本的位置,对于宏,参数名被他们的值所替换
  3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号,如果有,就重复上述过程。

注意:

宏参数和#define定义中可以出现其他#define定义的符号,但是对于宏,不能出现递归

当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

宏函数的对比

宏通常被应用于执行简单的运算

比如在两个数中找到比较大的一个时,写成下面的宏,更有优势一些。

#define MAX(a,b) ((a)>(b)?(a):(b))

那为什么不用函数来完成这个任务?

原因有2:

  1. 用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多,所以宏比函数在程序的规模和速度方面更胜一筹。
  2. 更为重要的是函数的参数必须声明为特定的类型,所以函数只能在类型合适的表达式上使用,反之这个宏可以适用于整型,长整型,浮点型等可以用于>来比较的类型,宏是类型无关的。

和函数相比宏的劣势:

  1. 每次使用宏的时候,一份宏的定义的代码插入到程序中,除非宏比较短,否则会增加代码的长度。
  2. 宏时没法调试的
  3. 宏由于类型无关,所以不够严谨
  4. 宏可能会带来运算符优先级的问题,导致程序容易出错

宏有时候可以做函数做不到的事情,比如,宏的参数可以出现类型,但是函数做不到

#define MALLOC(num,type) \
(type*)malloc(num,sizeof(type))
MALLOC(5.int)

 #和##

#运算符

#运算符将宏的一个参数转换为字符串字面量,它仅允许出现在带参数的宏的替换列表中。

#运算符所执行的操作可以理解为“字符串化”

当我们有一个变量 int a=10;的时候,我们想打印 :the value of a is 10

就可以写

#define PRINT(n) printf("the value of "#n"is %d",n)

当我们按照下面的方式调用的时候:

PRINT(a);当我们把a替换到宏的体内时,就出现了#a,而#a就是转换为‘a’时一个字符串代码就会预处理为:

printf("the value of 'a'is %d", a);

运行代码就能在屏幕上打印:

the value of a is 10

##运算符

## 可以把位于它两边的符号合成一个符号,它允许宏定义从分离的文本片段创建标识符。##被称为记号粘合

这样的链接必须产生一个合法的标识符,否则其结果就是未定义的

这里我们想想,写一个函数求2的数的最大值的时候,不同的数据类型就得写不同的函数

比如:

int int_max(int x, int y)
{return x > y ? x : y;
}
float float_max(float x, float y)
{return x > y ? x : y;
}

但是这样写起来太繁琐了,现在我们这样写代码试试:

//宏定义
#define GENERIC_MAX(type) \
type type##_max(type x,type y)\
{\
return(x > y ? x : y);\
}

使用宏,定义不同函数

命名约定

 一般来讲函数的宏的使用语法很相似,所以语言本身没法帮我们区分二者,那我们平时的一个习惯是:

把宏名全部大写

函数名不要全部大写

#undef

这条指令用于移除一个宏定义

#undef NAME
//如果现存的一个名字需要被重新定义,那么它的旧名字需要首先被移除

 

这篇关于C语言系列-带有副作用的宏参数#和##命名约定宏替换的规则的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Python之变量命名规则详解

《Python之变量命名规则详解》Python变量命名需遵守语法规范(字母开头、不使用关键字),遵循三要(自解释、明确功能)和三不要(避免缩写、语法错误、滥用下划线)原则,确保代码易读易维护... 目录1. 硬性规则2. “三要” 原则2.1. 要体现变量的 “实际作用”,拒绝 “无意义命名”2.2. 要让

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型

深入浅出Java中的Happens-Before核心规则

《深入浅出Java中的Happens-Before核心规则》本文解析Java内存模型中的Happens-Before原则,解释其定义、核心规则及实际应用,帮助理解多线程可见性与有序性问题,掌握并发编程... 目录前言一、Happens-Before是什么?为什么需要它?1.1 从一个问题说起1.2 Haht

python语言中的常用容器(集合)示例详解

《python语言中的常用容器(集合)示例详解》Python集合是一种无序且不重复的数据容器,它可以存储任意类型的对象,包括数字、字符串、元组等,下面:本文主要介绍python语言中常用容器(集合... 目录1.核心内置容器1. 列表2. 元组3. 集合4. 冻结集合5. 字典2.collections模块

Python进行word模板内容替换的实现示例

《Python进行word模板内容替换的实现示例》本文介绍了使用Python自动化处理Word模板文档的常用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录技术背景与需求场景核心工具库介绍1.获取你的word模板内容2.正常文本内容的替换3.表格内容的

MySQL批量替换数据库字符集的实用方法(附详细代码)

《MySQL批量替换数据库字符集的实用方法(附详细代码)》当需要修改数据库编码和字符集时,通常需要对其下属的所有表及表中所有字段进行修改,下面:本文主要介绍MySQL批量替换数据库字符集的实用方法... 目录前言为什么要批量修改字符集?整体脚本脚本逻辑解析1. 设置目标参数2. 生成修改表默认字符集的语句3

基于Go语言开发一个 IP 归属地查询接口工具

《基于Go语言开发一个IP归属地查询接口工具》在日常开发中,IP地址归属地查询是一个常见需求,本文将带大家使用Go语言快速开发一个IP归属地查询接口服务,有需要的小伙伴可以了解下... 目录功能目标技术栈项目结构核心代码(main.go)使用方法扩展功能总结在日常开发中,IP 地址归属地查询是一个常见需求: