C/C++基础:宏

C/C++基础:宏

  • 简述
  • 宏的简单使用
      • 基础语法
      • 带参宏(宏函数)
      • 宏参字符串化#
      • 宏拼接##
  • 宏的陷阱
      • 多行定义
      • 宏中的空格
      • 宏函数不是函数
      • 行末分号问题
        • 一些建议
  • 宏的奇妙使用

简述

宏作为C/C++最有特色的语言性质之一,犹如魔法一般,合理的使用可以极大的提高开发效率。

宏(Macro) 是C/C++的一个预处理指令,本质上是编译开始前进行的简单文本替换,我们可以定义一组宏代码片段,在程序中多次使用, 从而减少开发时间和精力。甚至我们可以使用宏来实现一些奇怪的操作,例如在C语言中利用宏做到如同模版一样的泛型编程。

宏的简单使用

基础语法

宏的基础使用方式如下:

#define 标识符 替换列表

比如,我们可以定义一个简单的宏PII,将其替换为浮点型字面量3.14

#define PII 3.14// 我们可以在代码里使用PII这个宏
int main(){std::cout << PII << std::endl;		// PII 被展开为 3.14return 0;
}

带参宏(宏函数)

我们同样可以定义一个宏函数,可以让我们像普通的函数一样来调用他,其写法和最普通的文本替换宏无太大区别,参数和函数一样用括号包裹,用逗号(,)分隔即可,如下我们编写一个宏实现乘法的功能。

#define mul(a, b) a*bint main(){std::cout << mul(2, 3) << std::endl;	// mul(2, 3)展开为 2*3return 0;
}

宏参字符串化#

有些时候,我们希望我们往宏传递的参数可以以字符串的形式展开,这个时候我们可以替换列表里使用#来将参数转为字符串。你可能觉得有些抽象,我保证看完下面这个例子,你就会知道什么是宏参字符串化:

#define str1(x) #x
#define str2(x) xint main() {const char* s1 = str1(hello);		// 本行代码展开为 const char* s1 = "hello"// const char* s2 = str2(hello);	// 错误的代码,本行代码展开为 const char* s2 = helloconst char* s3 = str2("hello");		// 本行代码展开为 const char* s2 = "hello"return 0;
}

宏拼接##

我们经常会有这样的需求,我们需要定义很多操作类似的函数,仅仅函数的前缀名不同,比如你可能需要在某处声明一系列这样的函数:

void func_1(int a, int b);
void func_2(int a, int b);
...
void func_n(int a, int b);

如果参数列表非常长的话,重复写这么多相似的函数声明无疑是难受的,你也许会想到利用宏来简化这个问题:

// 注意这是错误的实现
#define XX(n) void func_ n (int a, int b)int main(){XX(1);XX(2);...XX(n);
}

然而代码并没有像我们希望的一样展开,因为 func_ 和 n 被空格分开了,并没有正确拼接在一起, 我们可以通过##将参数和其他东西拼接起来,如下:

#define XX(n) void func_ ## n (int a, int b)int main(){XX(1);		//展开为 void func_1(int a, int b)XX(2);...XX(n);
}

宏的陷阱

看完上面的内容,恭喜你,基本已经学会了宏的语法.但是在你实际运用这些宏之前,还需要了解宏常见的陷阱,防止写出劣质乃至错误的代码.

多行定义

如果你是一个使用宏的新手,且习惯于使用函数,那么你是很有可能写出以下这样的宏的:

#define max(a, b){(a > b) ? a : b
}int main(){int c = max(1, 3);std::cout << c << std::endl;return 0;
}

然而当你兴致勃勃的写完代码,编译,发现编译器无情的报错了,你可以改成一行来保证正确性,如下:

#define max(a, b){(a > b) ? a : b}

但是如果这个宏的替换列表很长呢?写在一行未免过于臃肿了,其实我们通过换行符\来进行换行,如下:

// 这是正确的实现
#define max(a, b){		\(a > b) ? a : b		\
}
// 注意最后一行是不需要\的int main(){int c = max(1, 3);std::cout << c << std::endl;return 0;
}

宏中的空格

你也许会觉得,一个空格有什么大不了的,那么请你看下面的这个例子:

#define max (a, b){	\
(a > b) ? a : b		\
}int main() {int c = max(1, 3);std::cout << c << std::endl;return 0;
}/*
max(1, 3)展开如下
(a, b){
(a>b)? a: b
}(1,3)
/*

发现区别了吗?没错,我们在定义宏的时候,max和(a, b)之间不小心多写了一个空格,这会导致max(1, 3)完全展开为不同的结果.然而,在定义函数的时候,我们将函数名和()之间添加一个空格是完全无影响的,宏则需要完全避免这类事情。

宏函数不是函数

观察下面的简单宏函数和普通函数:

#define mul_m(a, b) a*bint mul_f(int a, int b){return a * b;
}

你觉得这个宏可以完全替代函数的功能吗?观察以下的代码:

#define mul(a, b) a*bint main(){std::cout << mul(2 + 1, 3) << std::endl;return 0;
}

看出问题了吗?mul(2+1, 3)被展开为了2 + 1 * 3 ,由于运算符的优先级问题,我们得到了期望外的结果,如何避免这种问题呢?其实只要加上括号来保证优先级即可,如下:

#define mul(a, b) ((a) * (b))

这下看起来终于完美了!
还是说,并没有?考虑下面这个代码,思考为什么这个代码和我们的预期不太一样:

#define mul(a) ((a) * (a))int main(){int a = 2;std::cout << mul(++a)<< std::endl;return 0;
}

相信通过文本展开你已经发现了,没错,mul(++a)被展开后出现了两个++a,这不是我们期望的,所以我们并不能盲目的用宏来替换函数。
实际上,宏还不能像函数那样进行自递归定义,如下的这样的宏是错误的:

#define A(x) 3
#define B(x) A(x) + B(x) + C(x)
#define C(x) A(x) + B(x)

行末分号问题

考虑下面的代码:

#define print(str) printf(#str);int main() {if (true) print(hello);elseprint(world);return 0;
}

你能发现问题吗?问题其实出在第一个print,这里展开后printf(“hello”)后面有两个分号,导致else与if无法进行匹配了。这里有这许多的解决方案,我们很容易保证分号的数量不出错。

一些建议

事实上,对于一部分宏函数我们总是建议可以利用do{}while(0)语句包裹起来,这样可以很好的处理宏的一些副作用,如下:

#define print(str) do {printf(#str)} while(0)

这样不仅保证了宏内的生命周期,还强制调用宏的时候需要像函数调用一样,在其后加上分号(考虑直接使用{} 包裹的场景)。

宏的奇妙使用

你现在已经学完了宏的使用方式,那么如何实现文章开头提到的泛型编程呢?其实十分简单,实现如下:

#include <iostream>
#define mySwap(T, a, b) do{ T tmp = a; a = b; b=tmp;}while(0)int main() {int a = 1, b = 0;mySwap(int, a, b);std::cout << a << ' ' << b << std::endl;return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

研究生选择学习Android开发的利与弊?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「Android的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;产品经理可以学学Axure快…

简单的CSS样式

样式分为三种 内部样式&#xff1a;写在html文件里的样式叫内部样式 内联样式&#xff1a;写在需要的标签中 外部样式&#xff1a;写在外部css文件里 可以通过不同的选择器来选择设置指定组件的样式&#xff1a; <style>/* 写在html文件里的样式叫内部样式 *//* 选择器 *…

6.3 面向对象技术-设计模式

设计模式 创建型模式 结构型模式

C++ unordered_map

1. unordered系列关联式容器 在C98 中&#xff0c; STL 提供了底层为红黑树结构的一系列关联式容器&#xff0c;在查询时效率可达到 &#xff0c;即最差情况下需要比较红黑树的高度次&#xff0c;当树中的节点非常多时&#xff0c;查询效率也不理想。最好的查询是&#xff0c…

我在百科荣创企业实践——简易函数信号发生器(6)

对于高职教师来说,必不可少的一个任务就是参加企业实践。这个暑假,本人也没闲着,报名参加了上海市电子信息类教师企业实践。7月8日到13日,有幸来到美丽的泉城济南,远离了上海的酷暑,走进了百科荣创科技发展有限公司。在这短短的一周时间里,我结合自己的教学经验和企业的…

Leetcode 剑指 Offer II 088.使用最小花费爬楼梯

题目难度: 简单 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 数组的每个下标作为一个阶梯&#xff0c;第 i 个阶梯对应着一个非…

5 Java的基本程序设计结构(基本语法4)- 集合之ArryList 和什么是包装类?

文章目录 前言一、集合二、基本数据类型的包装类三、ArryList1 ArryList对象的创建2 ArryList常见成员方法(1)boolean add(E e) : 添加元素,返回值表示是否添加成功(2)void add(int index, E e) :在指定索引位置插入元素。(3)boolean remove(E e) : 删除第一个指定元素…

从json到protobuf,接口效率的提升

在express开发的前后端调用中&#xff0c;express作为接收方是不二之选&#xff0c;它有一些很好用的body解析器来解析传入数据&#xff1b;而作为请求发起方&#xff0c;axios是非常方便的&#xff0c;这是一个很好的选择&#xff0c;它可以传输多种类型的数据给接收方。 通常…

Tekion 选择 ClickHouse Cloud 提升应用性能和指标监控

本文字数&#xff1a;4187&#xff1b;估计阅读时间&#xff1a;11 分钟 作者&#xff1a;ClickHouse team 本文在公众号【ClickHouseInc】首发 Tekion 由前 Tesla CIO Jay Vijayan 于 2016 年创立&#xff0c;利用大数据、人工智能和物联网等技术&#xff0c;为其汽车客户解决…

Week 3 DAY 6

Product C - Product (atcoder.jp) 一共N层&#xff0c;对于每一层的每个数&#xff0c;都遍历上一层更新过后的结果&#xff0c;更新为新的结果&#xff0c; 比如样例&#xff1a; 2 40 3 1 8 4 2 10 5动态数组a表示储存上一层除后留下来的数&#xff0c; 第一次a数组中只…

关于开源项目分享的通知

后续会逐步分享更多好用的开源项目&#xff0c;加入圈子&#xff1a; 圈子加入https://pc.fenchuan8.com/#/index?forum86631&yqm5EV39扫码加入&#xff1a;

初识git工具~~上传代码到gitee仓库的方法

目录 1.背景~~其安装 2.gitee介绍 2.1新建仓库 2.2进行相关配置 3.拉取仓库 4.服务器操作 4.1克隆操作 4.2查看本地仓库 4.3代码拖到本地仓库 4.4关于git三板斧介绍 4.4.1add操作 4.4.2commit操作 4.4.3push操作 5.一些其他说明 5.1.ignore说明 5.2git log命令 …

日拱一卒 | JVM

文章目录 什么是JVM&#xff1f;JVM的组成JVM的大致工作流程JVM的内存模型 什么是JVM&#xff1f; 我们知道Java面试&#xff0c;只要你的简历上写了了解JVM&#xff0c;那么你就必然会被问到以下问题&#xff1a; 什么是JVM&#xff1f;简单说一下JVM的内存模型&#xff1f;…

电脑系统安装软件,让系统安装变得更简单。

电脑原版操作系统下载&#xff1a;MSDN系统库 电脑U盘装机pe系统&#xff1a;优启通或微PE工具 驱动安装&#xff1a;360 驱动大师 电脑装机常用软件下载&#xff1a;https://www.bgrdh.com/favorites/7875.html

do while打印1~10

#include<stdio.h> int main() {int i 1;do{printf("%d", i);i;} while (i < 10);return 0; }

【JUC】LockSupport线程等待唤醒

文章目录 LockSupport线程等待唤醒机制三种让线程等待和唤醒的方法Object类中的wait和notify方法实现线程等待和唤醒Condition接口中的await和signal方法实现线程的等待和唤醒上述两种方法使用限制条件LockSupport类中的park等待和unpark唤醒LockSupport 是什么主要方法代码测试…

网易云音乐黑胶VIP会员免费领取入口直达词令是什么?

网易云音乐黑胶VIP会员免费领取是指网易云音乐VIP会员根据不同的等级尊享不同的权益&#xff0c;其中赠送礼品卡就是其一。不同等级的网易云音乐VIP会员可赠送的7天黑胶VIP会员张数不同&#xff0c;但是由于数量有限&#xff0c;每次更新后先领先得&#xff0c;我们将不定期根据…

SpringBoot3:轻松使用Jasypt实现配置文件信息加密

文章目录 前言一、概述1.1 Jasypt库简介1.2 Jasypt库的主要特点 二、开发环境三、Jasypt集成到SpringBoot33.1 引入依赖3.2 配置Jasypt3.3 加密配置文件信息3.3.1 方案一&#xff08;不推荐&#xff09;a.编写测试类生成加密后的配置文件信息b.运行c.修改原本的配置文件信息 3.…

vue实现电子签名、图片合成、及预览功能

业务功能&#xff1a;电子签名、图片合成、及预览功能 业务背景&#xff1a;需求说想要实现一个电子签名&#xff0c;然后需要提供一个预览的功能&#xff0c;可以查看签完名之后的完整效果。 需求探讨&#xff1a;后端大佬跟我说&#xff0c;文档我返回给你一个PDF的oss链接…

开源大模型的格式转成GGUF,并量化后使用ollama推理

https://github.com/ggerganov/llama.cpphttps://github.com/ggerganov/llama.cpp使用到的工具: llama.cpp ollama 步骤 1、下载llama.cpp,并使用make编译 2、新建conda环境,安装llama.cpp里所需的库(requirements.txt) 3、下载需要量化的模型