跟我学C++中级篇——if constexpr的应用

一、场景应用
在一个开发场景下,需要动态处理不同类型的数据写入。本来这个非常简单,只要定义一个模板即可搞定,但这里偏偏有一个细节,是调用别人的库来实现写入。而这个库对不同的数据类型的写入,提供了N种不同的函数,这些函数只是名字和写入数据类型不同,其它都完全一样。象下面的样子:

void putIntData(int *buf,int count,...);
void putCharData(char*buf,int count,...);
void putFloatData(float*buf,int count,...)
......

如果在上层应用调用这些函数,如下:

int * buf[1024] = {......};
int size = 1024;void SaveData(int *buf,uint32_t size)
{if(std::is_same_v<decltype(buf[0]),int>){putIntData(buf,size,...);}else if(std::is_same_v<decltype(buf[0]),char>){putIntData(buf,size,...);}else{}
}

这样的话编译无法通过,会报类型转换的错误。可上层的给定的缓冲区内的数据类型确实是可变的,做为一个中间处理层,如何能够正确的引导程序自动适配准确的函数调用呢?很容易想到使用模板。但是单纯的使用模板,仍然会报上面的错误。这也提醒,应该在编译期处理这个逻辑,理论上就会没有问题了。

二、分析
既然使用编译期来定位函数的调用,首先想到的使用用模板的特化来处理:

template <typename T> bool SaveDataSecond(T *buf, int size) {SaveCharData(buf, size);return true;
}
template <> bool SaveDataSecond(int *buf, int size) {SaveIntData(buf, size);return true;
}
template <> bool SaveDataSecond(float *buf, int size) {SaveFloatData(buf, size);return true;
}
int main() {// template specializationSaveDataSecond(buf, size);SaveDataSecond(fbuf, size);return 0;
}

然后可以想到学过的SFINAE,先考虑一下SFINAE的实现,最先想到的是std::enable_if系列:

#include <iostream>int buf[1024] = {0};
int size = 1024;float fbuf[1024] = {0.f};void SaveIntData(int *buf, int size, bool used = false) { std::cout << "save int type buffer!" << std::endl; }
void SaveCharData(char *buf, int size, bool used = false) { std::cout << "save char type buffer!" << std::endl; }
void SaveFloatData(float *buf, int size, bool used = false) { std::cout << "save float type buffer!" << std::endl; }template <typename T> std::enable_if_t<std::is_integral<T>::value, bool> SaveData(T *buf, int size) {SaveIntData(buf, size);return 0;
}
template <typename T> std::enable_if_t<std::is_floating_point<T>::value, bool> SaveData(T *buf, int size) {SaveFloatData(buf, size);return 0;
}
// void SaveData(int *buf, int size) {}
int main() {SaveData(buf, size);SaveData(fbuf, size);return 0;
}

看到这些代码是不是想到了std::is_same系列,即把is_floating_point等替换为std::is_same,如下:

template <typename T> std::enable_if_t<std::is_same_v<T, float>, bool> SaveData(T *buf, int size) {SaveFloatData(buf, size);return 0;
}

然后就是考虑一下在C++高版本(C++17及以后)有没有更好的解决办法即C++17后的if constexpr,如下面的代码用法:

   if constexpr(std::is_same<T,int>::value){//call switch}

三、解决
最后的解决办法就是使用if constexpr在编译期处理函数的分支调用:

int * buf[1024] = {......};
int size = 1024;template<typename T>
void SaveData(T*buf,uint32_t size)
{if constexpr(std::is_same_v<T,int>){putIntData(buf,size,...);}else if constexpr(std::is_same_v<T,char>){putIntData(buf,size,...);}else{}
}

这种方法实现起来既简单又容易理解,类似的问题,都可以使用这种方式来处理。

四、总结
技术的前进一般是迭代推进的。完全全新的知识点,往往很少。要关于对老的知识点的综合应用并不断的总结这种用法的可用之处,从而不断的推导出问题的解决办法。再通过这种解决办法可以看新的标准中是否有类似的更方便的方法,就可以更好的理解和认知一些技术点。从而可以更好的更深入的掌握它。与诸君共勉!

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

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

相关文章

Python实战开发及案例分析(28)—— 预编码算法

预编码算法&#xff08;Precoding Algorithm&#xff09;通常用于无线通信系统中&#xff0c;尤其是多输入多输出&#xff08;MIMO&#xff09;系统中&#xff0c;以提高数据传输的可靠性和效率。预编码是为了在发送端对信号进行处理&#xff0c;以优化传输性能。 在MIMO系统中…

Java设计模式 _行为型模式_访问者模式

一、访问者模式 1、访问者模式 访问者模式&#xff08;Visitor Pattern&#xff09;是一种行为型模式。它允许在不修改已有类结构的情况下&#xff0c;向类中添加新的操作。访问者模式通过将操作封装在一个访问者对象中&#xff0c;使得可以在不改变各个元素类的前提下&#x…

RedisTemplate实战应用--队列等

一、RedisTemplate队列插入 1、从集合左边插入值 https://blog.csdn.net/weixin_43658899/article/details/121040307 leftPush(K key, V value) redisTemplate.opsForList().leftPush("leftdatakey","bbbb");2、从集合左边开始在v1值后边插入新值v2 le…

使用 Django 连接 MySQL 数据库

文章目录 步骤一&#xff1a;安装必要的库和驱动步骤二&#xff1a;配置数据库连接步骤三&#xff1a;执行数据库迁移步骤四&#xff1a;开始使用 MySQL 数据库创建一个模型迁移模型到数据库使用模型进行数据操作创建新记录&#xff1a;查询记录&#xff1a;更新记录&#xff1…

Mac安装第三方软件的命令安装方式

场景&#xff1a; 打开终端命令行&#xff0c;sudo xattr -rd com.apple.quarantine&#xff0c;注意最后quarantine 后面加一个空格&#xff01;然后打开Finder&#xff08;访达&#xff09;&#xff0c;点击左侧的 应用程序&#xff0c;找到相关应用&#xff0c;拖进终端qua…

(超实用)京东订单数据分析案例-维度下钻

1&#xff0c;数据介绍&#xff0c;字段了解 尽可能熟悉业务&#xff0c;多知道字段的含义&#xff0c;字段字段间的逻辑关系&#xff0c;后期数据分析思路才能更清晰&#xff0c;结果才能更准确 2&#xff0c;订单数据分析基本思路 维度下钻 3&#xff0c;代码实现全流程思路…

华为telnet的两种认证方式

华为telnet的两种认证方式 实验拓扑&#xff1a; 实验要求&#xff1a; 1.采用普通密码认证实现telnet 远程登录机房设备R3 2.采用AAA认证服务方式实现telnet 远程登录机房设备R3 实验步骤&#xff1a; 1.完成基本配置&#xff08;设备接口配置IP&#xff0c;此步骤略过&#…

Facebook的隐私保护挑战:用户数据安全的新时代

在全球范围内&#xff0c;Facebook已经成为了不可忽视的社交媒体巨头&#xff0c;它连接着超过20亿的活跃用户。然而&#xff0c;随着其影响力的不断扩大&#xff0c;关于用户隐私和数据安全的问题也愈加引人关注。本文将深入探讨Facebook面临的隐私保护挑战&#xff0c;以及它…

一个程序员的牢狱生涯(47)学法

星期一 学法 二铺不知道什么时候走到了我的身边,向我说道,这是二铺在我进来号子后主动过来和我说话。 我听到二铺这声突兀的说话后,抬起头。这时我才看到,除了二铺,还有六子、棍子都围在我的身边,看着我。虽然六子和棍子依旧一副‘吊儿郎当’的样子,但我从他们几个的眼神…

解析前端开发中同源策略与配置代理

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 在前端开发中&#xff0c;跨域请求是一个常见的问题。同源策略限制了浏览器中一个页面…

C++高手进阶:Windows 模块加载的艺术与策略

前文我们讲到了怎么不依赖第三库&#xff0c;搭建自己的测试框架 没有看的读者可以通过这个链接自行阅读&#xff1a; &#x1f449;&#x1f449;&#x1f449; 自力更生&#xff1a;0依赖三方库&#xff0c;手把手教你打造专属C测试框架 作为项目开发来说&#xff0c;我们通常…

Leetcode:最长回文子串

题目链接&#xff1a;5. 最长回文子串 - 力扣&#xff08;LeetCode&#xff09; 普通版本&#xff08;暴力枚举&#xff09; 解题关键&#xff1a; 1、记录最长回文字串的长度和起始字符的下标 2、判断回文字串的逻辑与整体逻辑分离 3、先确定寻找回文字串的边界范围后从两边向…

解析Java中1000个常用类:CharSequence类,你学会了吗?

在 Java 编程中,字符串操作是最常见的任务之一。为了提供一种灵活且统一的方式来处理不同类型的字符序列,Java 引入了 CharSequence 接口。 通过实现 CharSequence 接口,各种字符序列类可以提供一致的 API,增强了代码的灵活性和可扩展性。 本文将深入探讨 CharSequence 接…

NBM 算法【python,算法,机器学习】

朴素贝叶斯法&#xff08;Naive Bayes model&#xff09;是基于贝叶斯定理与特征条件独立假设的分类方法。 贝叶斯定理 P ( A ∣ B ) P ( B ∣ A ) ∗ P ( A ) P ( B ) P(A|B)\frac{P(B|A) * P(A)}{P(B)} P(A∣B)P(B)P(B∣A)∗P(A)​ 其中A表示分类&#xff0c;B表示属性&…

Unity中的MVC框架

基本概念 MVC全名是Model View Controller 是模型(model)-视图(view)-控制器(controller)的缩写 是一种软件设计规范&#xff0c;用一种业务逻辑、数据、界面显示 分离的方法组织代码 将业务逻辑聚集到一个部件里面&#xff0c;在改进和个性化定制界面及用户交互的同时&#x…

【嵌入式硬件】DRV8874电机驱动

目录 1 芯片介绍 1.1 特性简介 1.2 引脚配置 1.3 最佳运行条件 2 详细说明 2.1 PMODE配置控制模式 2.1.1 PH/EN 控制模式 2.1.2 PWM 控制模式 2.1.3 独立半桥控制模式 2.2 电流感测和调节 2.2.1 IPROPI电流感测 2.2.2 IMODE电流调节 3.应用 3.1设计要求 3.2 设计…

AI换脸FaceFusion一键云部署指南

大家好&#xff0c;从我开始分享到现在&#xff0c;收到很多朋友的反馈说配置很低玩不了AI。本篇是一个云端部署AI项目的指南&#xff0c;帮助大家在云端进行AI项目的部署。我会从云平台的选择、代码部署、保存镜像几个方面进行详细的介绍。没有代码基础的小白也不用担心&#…

exe4j innosetup

exe4j:jdk: 打包时&#xff1a;需要的文件最好放到单独的一个文件夹下&#xff0c;主机安装32位jdk,exe4j用32位的。 附带jre: jre用32位的&#xff08;jdk下的jre&#xff09;可使用X86,X64.用相对路径。 只打64位时&#xff0c;需要选择32-bit or 64-bit (generate 64…

乐观锁 or 悲观锁 你怎么选?

你有没有听过这样一句话&#xff1a;悲观者正确&#xff0c;乐观者成功​。那么今天我来分享下什么是乐观锁​和悲观锁。 乐观锁和悲观锁有什么区别&#xff0c;它们什么场景会用 乐观锁 乐观锁基于这样的假设&#xff1a;多个事务在同一时间对同一数据对象进行操作的可能性很…

fps游戏中如何将矩阵转换为二维屏幕上的矩形坐标

fps游戏中如何将矩阵转换为二维屏幕上的矩形坐标 matrix[4][4]: 4x4 矩阵&#xff0c;通常用于3D变换&#xff08;如模型视图投影矩阵&#xff09;。 float* location: 一个指向位置坐标的指针&#xff0c;表示要转换的3D位置。 int Window_w, int Window_h: 窗口的宽度和高…