C++新经典模板与泛型编程:用成员函数重载实现std::is_convertible

news2025/1/13 10:27:02

用成员函数重载实现is_convertible

C++标准库中提供的可变参类模板std::is_convertible,这个类模板的主要能力是判断能否从某个类型隐式地转换到另一个类型,返回的是一个布尔值true或false。例如,一般的从int转换成float或从float转换成int,都是可以的。又如,有一个类A和一个类B,代码如下。

class A
{
};

class B : public A
{
};

因为类B的父类是类A,所以从类B转换到类A是可以的,但从类A转换到类B是不行的。
在main()主函数中加入几行测试代码:

#include "killCmake.h"

#include<string>

using namespace std;

class A
{
};

class B : public A
{
};


int main()
{
	// std::is_convertible<X,Y> :是判断能否从X类型转到Y类型
	std::cout << std::is_convertible<float, int>::value << std::endl;
	std::cout << std::is_convertible<int, float>::value << std::endl;

	// 子类转换到父类可以,但是父类不能转换到子类,会截断
	std::cout << std::is_convertible<A, B>::value << std::endl;
	std::cout << std::is_convertible<B, A>::value << std::endl;

	return 0;
}

在这里插入图片描述
从结果中可以看到,从类A转换到类B是不允许的(结果为0),其他几个转换都是可以的(结果为1)。
在明白了std::is_convertible的功能后,现在就来深入了解一下它的实现源码。这里,IsConvertibleHelper类模板和一个IsConvertible类模板,完成与std::is_convertible同样的功能。这里尤其注意,不要把类型模板的参数搞反,第1个类型模板参数叫作FROM,第2个类型模板参数叫作TO。IsConvertibleHelper类模板实现的功能是测试从FROM类型转换到TO类型是否可行。

template<typename From,typename To>
struct IsConvertibleHelper
{
private:
	static void testfunc(To);

	template<typename = decltype(testfunc(std::decltype<From>()))>
	static std::true_type test(void*);

	static std::false_type test(...);
public:
	using type = decltype(test(nullptr));
};
  • 上面的代码与以往讲过的IsDefConstructibleHelper代码非常类似,重载的test()成员函数返回类型分别是std::true_type和std::false_type。如果FROM类型能转换成TO类型,那么就会匹配返回std::true_type的test()成员函数(成员函数模板);否则会匹配返回std::false_type的test成员函数。
  • 值得注意的是,返回std::true_type的test()成员函数中类型模板参数默认值的写法,看起来是用decltype推断testfunc()成员函数的返回类型,传递给testfunc()的实参可以看作一个FROM类型的对象(std::declval()),如果FROM类型能被顺利地转换为TO类型,那么通过decltype推断testfunc()的返回类型的写法就是有效的(SFINAE原则),test()函数就会返回std::true_type,否则编辑器就会匹配返回类型为std::false_type的test()成员函数。
  • 现在,继续实现IsConvertible类模板,让其继承刚刚定义的IsConvertibleHelper模板中的type(type是一个类型,为std::true_type或std::false_type),代码如下。
#include "killCmake.h"

#include<string>

using namespace std;

class A
{
};

class B : public A
{
};

template<typename From,typename To>
struct IsConvertibleHelper
{
private:
	static void testfunc(To);

	template<typename = decltype(testfunc(std::decltype<From>()))>
	static std::true_type test(void*);

	static std::false_type test(...);
public:
	using type = decltype(test(nullptr));
};

template<typename From,typename To>
struct IsConvertible : IsConvertibleHelper<From, To>::type // struct 默认的继承方式是public继承
{
};

int main()
{
	// std::is_convertible<X,Y> :是判断能否从X类型转到Y类型
	std::cout << std::is_convertible<float, int>::value << std::endl;
	std::cout << std::is_convertible<int, float>::value << std::endl;

	// 子类转换到父类可以,但是父类不能转换到子类,会截断
	std::cout << std::is_convertible<A, B>::value << std::endl;
	std::cout << std::is_convertible<B, A>::value << std::endl;

	return 0;
}

在这里插入图片描述

  • 当FROM类型能够转换成TO类型时,IsConvertible的父类(IsConvertibleHelper<FROM, TO>::type)相当于std::true_type类型;当FROM类型不能转换成TO类型时,IsConvertible的父类就相当于std::false_type类型。此时,std::true_type或std::false_type类型中的静态成员变量value的值为true(1)或false(0)就代表FROM类型能否转换为TO类型。

可以用一个变量模板简化书写:

#include "killCmake.h"

#include<string>

using namespace std;

class A
{
};

class B : public A
{
};

template<typename From,typename To>
struct IsConvertibleHelper
{
private:
	static void testfunc(To);

	template<typename = decltype(testfunc(std::decltype<From>()))>
	static std::true_type test(void*);

	static std::false_type test(...);
public:
	using type = decltype(test(nullptr));
};

template<typename From,typename To>
struct IsConvertible : IsConvertibleHelper<From, To>::type // struct 默认的继承方式是public继承
{
};

template<typename From,typename To>
constexpr bool IsConvertible_v = IsConvertible<From, To>::value;


int main()
{

	std::cout << IsConvertible_v<float, int> << std::endl;
	std::cout << IsConvertible_v<int, float> << std::endl;
	std::cout << IsConvertible_v<A, B> << std::endl;
	std::cout << IsConvertible_v<B, A> << std::endl;

	return 0;
}

在这里插入图片描述

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

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

相关文章

相控阵天线(十四):常规大阵列天线分布(椭圆阵列、三角阵列、矩形拼接阵列、栅格拼接阵列)

目录 简介椭圆阵列三角阵列子阵拼接的矩形阵列子阵拼接的圆形阵列圆形子阵拼接阵列子阵栅格拼接阵列 简介 前面的博客已经介绍过常见的平面阵有一些基本类型&#xff0c;本篇博客介绍一些实际工程中可能出现的阵列&#xff0c;包括椭圆阵列、子阵通过矩形拼接形成的矩形大阵列…

题目:分糖果(蓝桥OJ 2928)

题目描述&#xff1a; 解题思路&#xff1a; 本题采用贪心思想 图解 题解&#xff1a; #include<bits/stdc.h> using namespace std;const int N 1e6 9; char s[N];//写字符串数组的一种方法,像数组一样***int main() {int n, x;cin >> n >> x;for(int …

有哪些已经上线的vue商城项目?

前言 下面是一些商城的项目&#xff0c;需要练手的同学可以挑选一些来练&#xff0c;废话少说&#xff0c;让我们直接开始正题~~ 1、newbee-mall-vue3-app 是一个基于 Vue 3 和 TypeScript 的电商前端项目&#xff0c;它是 newbee-mall 项目的升级版。该项目包含了商品列表、…

EXP-00056: 遇到 ORACLE 错误 12154 ORA-12154: TNS: 无法解析指定的连接标识符

exp oas/oasoas filed:\daochu.dmp owner(s) 导出特定用户 //exp 用户名/密码数据库 filed:\daochu.dmp owner(用户名) 1.重启oracle监听 cmd 中输入 services.msc 找到服务&#xff1a;OracleOraDb10g_home1TNSListener 与 OracleServiceORCL。 把两个服务启动. 若未解决…

CRM系统选择技巧,什么样的CRM系统好用?

SaaS行业发展迅速&#xff0c;更多的企业逐渐选择CRM管理系统。打开搜索引擎&#xff0c;有非常多的结果。怎样在数十万个搜索结果中选择适合您的CRM系统&#xff1f;下面我们将聊聊&#xff0c;怎样选择CRM系统。 第一步&#xff1a;明确自身需求 重要性&#xff1a;每家企业…

机器学习与低代码开发:创新驱动的双剑合璧

引言 随着科技的日新月异&#xff0c;机器学习和低代码开发已经成为引领技术行业变革的两大重要趋势。机器学习通过模拟人类的学习方式&#xff0c;让计算机具备了自我学习和预测的能力&#xff0c;打破了传统计算机程序的局限性。而低代码开发则以简化软件开发过程为目标&…

leetcode 1466

leetcode 1466 如图 node 4 -> node 0 -> node 1 因为节点数是n, 边长数量是n-1。所以如果是从0出发的路线&#xff0c;都需要修改&#xff0c;反之&#xff0c;如果是通向0的节点&#xff0c;例如节点4&#xff0c;则把节点4当作父节点的节点&#xff0c;之间的路线的方…

土壤水分传感器土壤体积含水率含量监测仪器

产品概述 外型小巧轻便&#xff0c;便于携带和连接。 土壤水分传感器由电源模块、变送模块、漂零及温度补偿模块、数据处理模块等组成。传感器内置信号采样及放大、漂零及温度补偿功能&#xff0c;用户接口简洁、方便。 功能特点 ◆本传感器体积小巧化设计&#xff0c;测量…

妙手ERP功能更新:TikTok支持打印10×10面单、Ozon支持设置最低售价、超过90天的Shopee订单买家信息不再显示......

为了给卖家朋友带来更好的使用体验&#xff0c;更高效地运营跨境店铺&#xff0c;妙手ERP在上周优化了以下多项功能。 01、产品模块优化 全平台 - 自定义平台SKU时&#xff0c;连接符支持为空或可输入其他符号&#xff08;不支持输入数字、emoji、文字&#xff09; Ozon - 支…

modbus转profinet网关解决plc插槽号不够用的情况

PLC作为常用的控制设备之一&#xff0c;其插槽号有时会限制外部设备的连接数量。然而&#xff0c;通过使用modbus转profinet网关&#xff0c;可以解决这一问题。这种设备能够将modbus协议转换为profinet协议&#xff0c;实现PLC与更多外部设备的连接。 modbus转profinet网关还具…

【ET8】1.ET8入门-运行指南

主要学习网址 论坛地址为&#xff1a;https://et-framework.cn Git地址为&#xff1a;GitHub - egametang/ET: Unity3D Client And C# Server Framework 官方QQ群 : 474643097 项目检出 检出项目切换到release8.0分支 GitHub地址&#xff1a;GitHub - egametang/ET: Unity…

大数据毕业设计之前端02:架构布局和aside的设计

前言 上一篇主要讲了我学习前端的一个经历&#xff0c;以及为什么选择BuildAdmin作为深入前端学习的原因.同事也大致聊了一下学习前端需要使用哪些技术栈。 本篇文章来拆解一下BuildAdmin的前端代码结构&#xff0c;和布局实现的细节。 前端代码结构 必须先了解项目的结构&…

无人机巡检中台揭秘,无人机管控平台助力巡查无人机巡检方案落地

随着无人机智能巡检的飞速发展&#xff0c;巡查无人机应用场景也日益多元&#xff0c;无人机巡检方案被频繁落地到工业生产及巡检巡逻业务中。而无人机管控平台应运而生&#xff0c;成为推动无人机智能巡检的关键工具。那我们一起来看一下无人机管控平台的作用&#xff1a; 一、…

为“异常”努力是值得的

异常是OO语言处理错误的方式,在C中&#xff0c;鼓励使用异常。侯捷再书中谈起异常&#xff0c;“十年前撰写“未将异常考虑在内的”函数是为一种美好实践&#xff0c;而今我们致力于写出“异常安全码”。”可见异常安全的重要。 说起异常安全&#xff0c;首先就要是异常的出现…

免费的网页抓取工具大全【2024最新】

在当今数字化时代&#xff0c;信息的获取变得愈发重要。企业、学者、研究人员都在努力寻找一种高效、可靠的方式从海量的互联网数据中提取有价值的信息。本文将深入探讨各种网页抓取工具的类型。 揭秘网页抓取工具的丰富多样 在网络数据抓取的领域&#xff0c;有多种工具可供…

校园网无法登录IEEE

校园网无法登录IEEE 取消IPV6协议版本的对勾

python数据分析总结(pyecharts)

目录 安装依赖 Demo pyecharts.options 图表绘制 WordCloud.shape 安装依赖 pip install pyecharts Demo ### 对象式 from pyecharts.charts import Bar bar Bar() bar.add_xaxis([1,2,3,4,5]) bar.add_yaxis("数据1",[40,56,65,23,56]) bar.add_yaxis("数…

CleanMyMac X清理垃圾软件有哪些功能?2024版本更新啥内容

如今电脑成为大多数人工作的工具&#xff0c;使用得越久就越需要清理垃圾软件。系统垃圾、废纸娄垃圾、大型和旧文件这些通常都占用了我们的电脑几G的空间。 想要清除不必要的垃圾文件&#xff0c;可以使用mac卸载软件CleanMyMac X&#xff0c;这款清理垃圾软件可以智能扫描Ma…

Linux查看命令的绝对路径

linux查看命令的绝对路径 在Linux中&#xff0c;可以使用以下命令来查看命令的绝对路径&#xff1a; 1、which 命令名 例如&#xff0c;要查看chronyc命令的绝对路径&#xff0c;可以运行&#xff1a; which chronyc 2、whereis 命令名 例如&#xff0c;要查看chronyc命令…

探究Logistic回归:用数学解释分类问题

文章目录 前言回归和分类Logistic回归线性回归Sigmoid函数把回归变成分类Logistic回归算法的数学推导Sigmoid函数与其他激活函数的比较 Logistic回归实例1. 数据预处理2. 模型定义3. 训练模型4. 结果可视化 结语 前言 当谈论当论及机器学习中的回归和分类问题时&#xff0c;很…