17 C 语言宏进阶必看:从宏替换避坑到宏函数用法,不定参数模拟实现一次搞定

 预处理详解

1. 预定义符号

//C语⾔设置了⼀些预定义符号,可以直接使⽤,预定义符号也是在预处理期间处理的。
__FILE__ //进⾏编译的源⽂件--预处理阶段被替换成指向文件名字符串的指针--char* 类型的变量
__LINE__ //⽂件当前的⾏号 --预处理阶段替换成使用该预处理符号所在的行号--unsigned int 类型
__DATE__ //⽂件被编译的⽇期
__TIME__ //⽂件被编译的时间
__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义

2 #define 定义常量

宏替换注意点:

1 只能替换在同一行的内容,如果内容太长可以在每一行结尾用 \表示绪行

2 宏替换本质上就是文本替换在预处理阶段就替换完成了,c语言代码叫编译型语言是因为在没有编译之前

代码就是文本文件,是可以进行替换的。替换之后再做语法检查

基本语法:

#define name stuff 1//这个只是宏替换的模板,后面使用到name字段按照这个模板替换成stuff 1

举个例⼦:

#define MAX 1000
#define reg register //为 register这个关键字,创建⼀个简短的名字
#define do_forever for(;;) //⽤更形象的符号来替换⼀种实现
#define CASE break;case //在写case语句的时候⾃动把 break写上。
// 如果定义的 stuff过⻓,可以分成⼏⾏写,除了最后⼀⾏外,每⾏的后⾯都加⼀个反斜杠(续⾏符)。
#define DEBUG_PRINT printf("file:%s\tline:%d\t \
date:%s\ttime:%s\n" ,\
__FILE__,__LINE__ , \
__DATE__,__TIME__ )
思考:在define定义标识符的时候,要不要在最后加上 ; ?
//⽐如:
#define MAX 1000;
#define MAX 1000//建议不要加上 ; ,这样容易导致问题。
//⽐如下⾯的场景:
if(condition)
max = MAX;
else
max = 0;//如果是加了分号的情况,等替换后,if和else之间就是2条语句,⽽没有⼤括号的时候,if后边只能有⼀
//条语句。这⾥会出现语法错误。

3 #define定义宏

#define机制包括了⼀个规定,允许把参数替换到⽂本中,这种实现通常称为宏(macro)或定义宏 下⾯是宏的申明⽅式:

#define name( parament-list ) stuff 

其中的 parament-list 是⼀个由逗号隔开的符号表,它们可能出现在stuff中。 注意: 参数列表的左括号必须与name紧邻,如果两者之间有任何空⽩存在,参数列表就会被解释为stuff的 ⼀部分。

举例:

#define SQUARE( x ) x * x

这个宏接收⼀个参数 x .如果在上述声明之后,你把 SQUARE( 5 ); 置于程序中,预处理器就会⽤下⾯这个表达式替换上⾯的表达式:

5 * 5

警告: 这个宏存在⼀个问题: 观察下⾯的代码段:

int a = 5;
printf("%d\n" ,SQUARE( a + 1) );

乍⼀看,你可能觉得这段代码将打印36,事实上它将打印11,为什么呢?? 替换⽂本时,参数x被替换成a+1,所以这条语句实际上变成了:

printf ("%d\n",a + 1 * a + 1 ); 

这样就⽐较清晰了,由替换产⽣的表达式并没有按照预想的次序进⾏求值。 在宏定义上加上两个括号,这个问题便轻松的解决了:

#define SQUARE(x) (x) * (x) 

这样预处理之后就产⽣了预期的效果:

printf ("%d\n",(a + 1) * (a + 1) );

4 宏替换的规则

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

  1. 在调⽤宏时,⾸先对参数进⾏检查,看看是否包含任何由#define定义的符号。如果是,它们⾸先 被替换。

  2. 替换⽂本随后被插⼊到程序中原来⽂本的位置。对于宏,参数名被他们的值所替换。

  3. 最后,再次对结果⽂件进⾏扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上 述处理过程。 注意:

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

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

5 宏函数和函数 的对⽐

注意点宏函数

宏函数的参数的替换是没有类型检查的,只是把参数替换进去最后编译才进行类型检查。是把宏参数做文本替换,替换掉后面的整个文本中的宏参数文本,比如:

#define MAX(a, b) ((a)>(b)?(a):(b)) int x=1,y=2;
//文本a和b替换成x,y,后面的文本中的a,b也被替换成x,y
MAX(x,y) ---替换成((x)>(y)?(x):(y)) 
//a,b-->1,2
MAX(1,2) ---替换成((1)>(2)?(1):(2)) 
//a,b -->1,2
MAX(1,x) --- ((1)>(x)?(1):(x)) 

宏通常被应⽤于执⾏简单的运算。 ⽐如在两个数中找出较⼤的⼀个时,写成下⾯的宏,更有优势⼀些。

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

那为什么不⽤函数来完成这个任务? 原因有⼆:

  1. ⽤于调⽤函数和从函数返回的代码可能⽐实际执⾏这个⼩型计算⼯作所需要的时间更多。所以宏⽐ 函数在程序的规模和速度⽅⾯更胜⼀筹。

  2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使⽤。反之 这个宏可以适⽤于整形、⻓整型、浮点型等可以⽤于 > 来⽐较的类型。宏的参数是类型⽆关的。 和函数相⽐宏的劣势:

  3. 每次使⽤宏的时候,⼀份宏定义的代码将插⼊到程序中。除⾮宏⽐较短,否则可能⼤幅度增加程序 的⻓度。

  4. 宏是没法调试的。

  5. 宏由于类型⽆关,也就不够严谨。

  6. 宏可能会带来运算符优先级的问题,导致程容易出现错。 宏有时候可以做函数做不到的事情。⽐如:宏的参数可以出现类型,但是函数做不到。


#define MALLOC(num, type)\
(type )malloc(num sizeof(type))
...
//使⽤
MALLOC(10, int); //类型作为参数
//预处理器替换之后:
(int * )malloc(10 sizeof(int));

6 #和##

1 #运算符

#运算符将宏的⼀个参数转换为字符串字⾯量。它仅允许出现在带参数的宏的替换列表中。 #运算符所执⾏的操作可以理解为”字符串化“。 当我们有⼀个变量 int a = 10; 的时候,我们想打印出: the value of a is 10 . 就可以写:

//首先把a文本替换掉宏函数中的n文本,然后在把后面的文本中的n文本也替换成a文本
//最后整体替换文本,宏函数替换成后面一整个文本
#define PRINT(n) printf("the value of "#n " is %d", n); 
int a=1;
PRINT(a);// the value of a is 10 .

当我们按照下⾯的⽅式调⽤的时候: PRINT(a);//当我们把a替换到宏的体内时,就出现了#a,⽽#a就是转换为 字符串"a",同时两个字符串是会被合并为一个字符串的,比如:

"abs""sad" ->"abssad"    "add"fff"dsa" --> "addfffdsa"

2 ##运算符

可以把位于它两边的符号合成⼀个符号,它允许宏定义从分离的⽂本⽚段创建标识符。 ## 被称为记号粘合 这样的连接必须产⽣⼀个合法的标识符。否则其结果就是未定义的。这⾥我们想想,写⼀个函数求2个数的较⼤值的时候,不同的数据类型就得写不同的函数。⽐如:

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

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

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

使⽤宏,定义不同函数 GENERIC_MAX(int) //替换到宏体内后int##max ⽣成了新的符号 int_max做函数名 GENERIC_MAX(float) //替换到宏体内后float##max ⽣成了新的符号 float_max做函数名


GENERIC_MAX(int) 
/*
预处理被文本替换成
int int_max(int x,int y)
{return ((x)>(y)?(x):(y));
}*/
GENERIC_MAX(float)
int main()
{
//调⽤函数
int m = int_max(2, 3);
printf("%d\n", m);
float fm = float_max(3.5f, 4.5f);
printf("%f\n", fm);
return 0;
}
输出:
3
4.500000
不定参数的传参使用##
#include<stdio.h>//__FILE__ :预处理符号,在预处理阶段替换成当前符号所在的文件名类型为字符串
//__LINE__ :预处理符号, 在预处理阶段替换成当前符号所在的文件的具体行数类型为int//不定宏参数的使用: ... 可变参数符号,不确定传入的参数是多少,在宏定义中通常搭配##__VA_ARGS__使用
//Print(fmt,...) 预处理替换时 Print(fmt,...)中的fmt 替换printf中的fmt ,"jjjj"替换 ...
//##__VA_ARGS__:若可变参数为空,C99标准要求逗号必须保留(可能引发语法错误)。
//GCC/Clang通过##__VA_ARGS__优化,展开时自动去掉前面的逗号
//传入Print(不定参数) 通过##__VA_ARGS__替换到printf函数中
//当多个字符串写在一起时编译预处理时会自动把多个字符串拼接在一起
//"ddd""dasd""asdsa" --> "ddddsdadsadsa"
#define Print(fmt,...) printf("[%s %d]"fmt"\n",__FILE__,__LINE__,##__VA_ARGS__)
int main()
{printf("[%s %d] %s\n",__FILE__,__LINE__,"jhhh");Print("%s","jjjj");Print("gggg");//传入的可变参数为空Print("ddd""fff");return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/95837.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/95837.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入剖析 HarmonyOS ArkUI 声明式开发:状态管理艺术与最佳实践

好的&#xff0c;请看这篇关于 HarmonyOS ArkUI 声明式开发范式与状态管理的技术文章。 深入剖析 HarmonyOS ArkUI 声明式开发&#xff1a;状态管理艺术与最佳实践 引言 随着 HarmonyOS 4、5 的广泛应用以及面向未来的 HarmonyOS NEXT&#xff08;API 12&#xff09;的发布&…

Qwen-Code安装教程

一、概述Qwen Code 是一个强大的基于命令行、面向开发者的 AI 工作流工具&#xff0c;改编自 Gemini CLI&#xff0c;专门针对 Qwen3-Coder 模型进行了优化。它专门为代码理解、代码重构、自动化工作流、Git 操作等场景设计&#xff0c;让你的开发工作变得更高效、更智能。它既…

老师傅一分钟精准判断电池好坏!就靠这个神器!

在汽车维修与保养领域&#xff0c;蓄电池状态的准确判断一直是技术人员面临的重要挑战。传统的电压测量方法只能反映表面现象&#xff0c;无法深入评估蓄电池的实际健康状态。Midtronics MDX-P300蓄电池及电气系统测试仪作为专业级诊断设备&#xff0c;通过电导测试技术和多系统…

Axure笔记

Axure介绍 快速原型的软件 应用场景&#xff1a;拉投资、给项目团队、销售演示、项目投标、内部收集反馈、教学 软件安装与汉化 汉化&#xff1a;复制lang文件夹和三个dll 软件的基础功能 基本布局&#xff1a;菜单栏、工具栏、页面和摘要、元件和母版、画布、样式交互和说明设…

Pytorch Yolov11 OBB 旋转框检测+window部署+推理封装 留贴记录

Pytorch Yolov11 OBB 旋转框检测window部署推理封装 留贴记录 上一章写了下【Pytorch Yolov11目标检测window部署推理封装 留贴记录】&#xff0c;这一章开一下YOLOV11 OBB旋转框检测相关的全流程&#xff0c;有些和上一章重复的地方我会简写&#xff0c;要两篇结合着看&#x…

《Keil 开发避坑指南:STM32 头文件加载异常与 RTE 配置问题全解决》

《Keil 开发避坑指南&#xff1a;STM32 头文件加载异常与 RTE 配置问题全解决》文章提纲一、引言• 简述 Keil 在 STM32 开发中的核心地位&#xff0c;指出头文件加载和 RTE&#xff08;运行时环境&#xff09;配置是新手常遇且关键的问题&#xff0c;说明本文旨在为开发者提…

TortoiseGit 2.4.0.0 64位安装教程(附详细步骤和Git配置 附安装包)

本教程详细讲解 ​TortoiseGit 2.4.0.0 64位版本​ 的完整安装步骤&#xff0c;包括如何运行 ​TortoiseGit-2.4.0.0-64bit.msi​ 安装包、设置安装路径、关联 Git 环境&#xff0c;以及安装后的基本配置方法&#xff0c;适合 Windows 用户快速上手 Git 图形化管理工具。 一、…

大数据毕业设计选题推荐-基于大数据的高级大豆农业数据分析与可视化系统-Hadoop-Spark-数据可视化-BigData

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、PHP、.NET、Node.js、GO、微信小程序、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇…

学习机器学习能看哪些书籍

关注B站可以观看更多实战教学视频&#xff1a;hallo128的个人空间 在机器学习与深度学习的知识海洋中&#xff0c;选择合适的书籍往往是入门和进阶的关键。以下四本经典著作各具特色&#xff0c;覆盖了从基础理论到实践应用的多个维度&#xff0c;无论你是初学者还是有一定基础…

Unity通过Object学习原型模式

原型模式简述 依据现有的实例生成新的实例 Object的实例化方法 Object.Instantiate 克隆 original 对象并返回克隆对象 Unity中的实例&#xff1a;预制体或场景中的游戏对象 示例 方法1&#xff1a;手动创建对象并添加组件 方法2&#xff1a;使用实例化方法&#xff0c;实…

【踩坑记录】Unity 项目中 PlasticSCM 掩蔽列表引发的 文件缺失问题排查与解决

问题描述&#xff1a; Plastic SCM 签入时&#xff0c;弹窗提示“项xxx在该工作区中不存在” Unity 项目中 PlasticSCM 掩蔽列表引发的 文件缺失问题排查与解决 文章目录Unity 项目中 PlasticSCM 掩蔽列表引发的 文件缺失问题排查与解决一、前言二、Unity 与 .meta 文件机制1. …

Redis实战-附近的人实现的解决方案

1.GEO数据结构1.1实现附近的人的数据结构Redis提供的专用的数据结构来实现附近的人的操作&#xff0c;这也是企业的主流解决方案&#xff0c;建议使用这种解决方案。GEO就是Redis提供的地理坐标计算的一个数据结构&#xff0c;可以很方便的计算出来两个地点的地理坐标&#xff…

HTML第七课:发展史

HTML第七课&#xff1a;发展史发展史快速学习平台发展史 示例 HTML 发展史 前端三件套&#xff1a;html 、css、javascript(Js) HTML 发展史 HTML 1.0&#xff08;1993 年&#xff09; 蒂姆伯纳斯 - 李&#xff08;Tim Berners - Lee&#xff09;发明了万维网&#xff0c;同…

中国生成式引擎优化(GEO)市场分析:领先企业格局与未来趋势分析

一、GEO市场变革中国生成式引擎优化&#xff08;Generative Engine Optimization, GEO&#xff09;市场正经历一场深刻的变革&#xff0c;其核心在于生成式人工智能&#xff08;Generative AI&#xff09;对传统搜索引擎和数字营销模式的颠覆性影响。传统搜索引擎以“提供链接”…

好看的背景颜色 uniapp+小程序

<view class"bg-decoration"><view class"circle-1"></view><view class"circle-2"></view><view class"circle-3"></view> </view>/* 背景装饰 */.container{background: linear-gr…

《驾驭云原生复杂性:隐性Bug的全链路防御体系构建》

容器、服务网格、动态配置等抽象层为系统赋予了弹性与效率,但也像深海中的暗礁,将技术风险隐藏在标准化的接口之下。那些困扰开发者的隐性Bug,往往并非源于底层技术的缺陷,而是对抽象层运行逻辑的理解偏差、配置与业务特性的错配,或是多组件交互时的协同失效。它们以“偶发…

vosk语音识别实战

一、简介 Vosk 是一个由 Alpha Cephei 团队开发的开源离线语音识别&#xff08;ASR&#xff09;工具包。它的核心优势在于完全离线运行和轻量级&#xff0c;使其非常适合在资源受限的环境、注重隐私的场景或需要低延迟的应用中使用。 二、核心特点 离线运行 (Offline) 这是…

鸿蒙ABC开发中的名称混淆与反射处理策略:安全与效率的平衡

在当今的软件开发中&#xff0c;代码安全是一个至关重要的议题。随着鸿蒙系统&#xff08;HarmonyOS&#xff09;的广泛应用&#xff0c;开发者们在追求功能实现的同时&#xff0c;也必须考虑如何保护代码不被轻易破解。名称混淆是一种常见的代码保护手段&#xff0c;但当反射机…

css页面顶部底部固定,中间自适应几种方法

以下是实现页面顶部和底部固定、中间内容自适应的几种常见方法&#xff0c;附代码示例和适用场景分析&#xff1a;方法一&#xff1a;Flexbox 弹性布局 <body style"margin:0; min-height:100vh; display:flex; flex-direction:column;"><header style"…

彻底拆解 CSS accent-color:一个属性,省下一堆“重造轮子”的苦工

我有一支技术全面、经验丰富的小型团队&#xff0c;专注高效交付中等规模外包项目&#xff0c;有需要外包项目的可以联系我既要原生控件、又要品牌配色&#xff0c;还不想伪造组件&#xff1f;能不能讲透 accent-color。下面给出一版尽量“到骨头里”的解析&#xff1b;对讨厌从…