C++:类型转换

news2024/11/7 5:26:29

目录

一. C语言的类型转换

二. C++类型转换

2.1 static_cast

2.2 reinterpret_cast

2.3 const_cast

2.4 dynamic_cast

三. 运行时类型识别 -- RTTI

四. 总结


一. C语言的类型转换

C语言的类型转换分为隐式类型转换和强制类型转换,隐式类型转换发生在相近的类型之间,强制类型转换可以完成不相关类型之间的转换。

  • 隐式类型转换:当运算符两侧的数据类型不同时,其中一个数据的类型会被隐式转换为另一个数据的类型,转换规则遵循图1.1中自上而下的顺序。
  • 强制类型转换:实现不相关类型之间的转换,如:将int型数据转换为int*类型的数据。编译器不会自动进行强制类型转换,需要用户显示的去进行转换。注意:并不是任何不相关的数据都能进行强制类型转换,如自定义类型和内置类型之间不能强转。
图1.1 隐式类型转换顺序

C语言的类型转换有以下缺陷:

  1. 隐式类型转换,有可能造成精度的丢失。
  2. 将所有强制类型转换混在一起,不易区分。 

为了弥补C语言类型转换的缺陷,C++提出了属于自己的四种类型转换方式,但是C++兼容C语言,C语言的类型转换规则在C++中依旧适用。

二. C++类型转换

C++有static_cast、interpret_cast、const_cast、_dynamic四种类型转换方式。

2.1 static_cast

用法:static_cast<转换后的类型>(被转换类型的变量)

功能:用于执行任何C语言中的隐式类型转换,不可以用于非相关类型的强制类型转换。

int main()
{
	double d1 = 12.1234;
	int i1 = static_cast<int>(d1);  //将d1的类型转换为int

	int i2 = 3;
	double d2 = static_cast<double>(i2);  //将i2的类型

	int a = 1234567;
	//int* pa = static_cast<int*>(a);   //无法实现不相关类型之间的强制类型转换

	return 0;
}

2.2 reinterpret_cast

用法:reinterpret_cast<转换后的类型>(被转换类型的变量)

功能:指向不相关类型之前的强制类型转换,不可用于相关类型之间的隐式类型转换

注意:如果被转换类型的对象具有const属性,那么使用reinterpret_cast不能取消其const属性,如:const int b = 10; int* pb = reinterpret_cast<int*>(&b)就无法通过编译,因为&b本身的类型为const int*, reinterpret_cast<int*>(&b)将其const属性取消了。

但是,使用C语言的强制类型转换(int*)&b能够取消const属性。

int main()
{
	int a = 1234567;
	int* pa = reinterpret_cast<int*>(a);
	std::cout << pa << std::endl;

	double d = 10.11;
	//int i = reinterpret_cast<int>(d);  //不能进行相近类型之间的转换

	const int b = 10;
	//int* pb1 = reinterpret_cast<int*>(&b);  //不能取消&b的const属性
	int* pb2 = (int*)&b;   //编译通过
	*pb2 = 20;

	return 0;
}

2.3 const_cast

用法:const_cast<不具有const属性的类型>(具有const属性的对象)

功能:取消原本对象/变量类型的const属性

如:const int a = 10,不能直接对a的值进行修改,但是,如果通过const_cast操作符,拿到不具有const属性的a的地址,就可以通过解引用改变a的值。

使用reinterpret_cast无法取消const属性。

int main()
{
	const int a = 1;
	int* pa = const_cast<int*>(&a);
	*pa = 2;
	return 0;
}

2.4 dynamic_cast

用法:dynamic_cast<子类指针或引用类型>(父类指针或引用)

功能:在保证安全的前提下,将父类的指针或引用类型转换为子类的指针或引用类型

  • 正常情况下,派生类对象可以赋值给基类对象,基类的对象一定不能赋值给派生类的对象。派生类的指针或引用可以赋值给基类的指针或引用。
  • 基类的指针或引用经过强制类型转换后,可以赋值给派生类的指针或引用,但这样并不安全,容易出现越界访问的问题。
  • dynamic_cast可以用于将基类对象的引用或指针,经强制类型转换后赋值给派生类的引用或指针。使用dynamic_cast要么类型转换失败,如果成功就一定安全。

关于dynamic_cast强制类型转换,有以下两点注意事项:

  1. 要求基类必须有虚函数,否则无法通过dynamic_cast实现类型转换。
  2. 如果强制将基类的引用转换为派生类的引用时失败,dynamic_cast会抛出异常,如果强制将基类的指针转换为派生类的指针时失败,dynamic_cast返回空指针nullptr。

一般来说,如果某个基类的引用或指针,实际指向某个派生类对象,dynamic_cast就会强转成功,如果基类的引用或指针就是指向某个基类对象而不是派生类对象,强转类型转换就会失败。

如下面的代码,定义了两个基类指针pa1和pa2,pa1实际指向基类对象a,pa2实际指向派生类对象b,将pa1和pa2分别作为参数传给func函数,func函数中尝试尝试使用dynamic_cast将它们强转类型转换为派生类指针,pa1转换失败,pa2转换成功。

class A
{
public:
	virtual void func() { }

public:
	int _a1 = 10;
};

class B : public A
{
public:
	int _b1 = 20;
};

void func(A* pa)
{
	//如果转换成功pb为正确的派生类对象地址
	//如果转换失败pb为nullptr
	B* pb = dynamic_cast<B*>(pa);

	//如果转换成功
	if (pb)
	{
		std::cout << "转换成功" << std::endl;
		std::cout << "_a1 = " << pb->_a1 << std::endl;
		std::cout << "_b1 = " << pb->_b1 << std::endl << std::endl;
	}
	else   //如果转换失败
	{
		std::cout << "转换失败" << std::endl;
		std::cout << "_a1 = " << pa->_a1 << std::endl << std::endl;
	}
}

int main()
{
	A a;
	B b;

	A* pa1 = &a;
	A* pa2 = &b;

	func(pa1);  //转换失败
	func(pa2);  //转换成功

	return 0;
}
图2.1 代码运行结果

三. 运行时类型识别 -- RTTI

C++通过以下三种方式,进行运行时类型失败:

  1. typeid -- 以字符串的形式获取变量/对象的类型
  2. dynamic_cast运算符 -- 检验基类的指针或引用能否转换为派生类的指针或引用。
  3. decltype关键字 -- 自动识别变量类型,可用其返回结果定义新的变量的类型。

四. 总结

  • C语言支持隐式类型转换和强制类型转换,隐式类型转换发生在相近类型之间,强制类型转换用于不相关类型之间的转换。
  • C++支持四种方式的类型转换,static_cast执行相近类型见的转换、reinterpret_cast用于不相关类型的强制转换,但不能取消const属性、const_cast用于取消对象类型的const属性、dynamic_cast用于将基类的指针或引用转换为派生类的指针或引用,dynamic_cast要求基类要有虚函数,否则会编译报错,dynamic_cast执行强制类型转换要么编译报错,如果转换成功就一定能保证安全。
  • C++通过typeid、dynamic_cast和decltype三种方式实现运行时类型识别。

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

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

相关文章

WEB测试环境搭建和测试方法大全

一、WEB测试环境搭建 WEB测试时搭建测试环境所需的软硬件包括&#xff1a;电脑一台、JDK1.6、Tomcat7.0、mysql、IE浏览器、Firefox浏览器、Chrome浏览器、SVN客户端 通过SVN客户端导出最新的Web工程部署到Tomcat7.0下的webapps中&#xff0c;另外重要的一点就是修改数据库连…

31、js - Promise

一、Promise要点 -> js中&#xff0c;只有Promise对象才可以使用.then().catch()方法。 -> axios可以使用.then().catch()&#xff0c;完全是因为调用axios()&#xff0c;返回的是一个Promise对象。 -> new Promise() 里面的代码是同步代码&#xff0c;一旦调用promis…

这个API Hub太厉害了,太适合接口测试了,收录了钉钉企业微信等开放Api的利器

目录 前言&#xff1a; 01API Hub的项目 02API Hub 03调试 04 API 调试 05 API mock 06 针对开放项目功提供者 08 下载 前言&#xff1a; API Hub 的优势在于它提供了完整的 API 管理解决方案&#xff0c;包括API的设计、接口调试、测试和文档管理等。通过集中管理API…

火热报名中 | KCD 北京精彩抢“鲜”看

​ 仲夏已至&#xff0c;风云再起&#xff0c;Kubernetes Community Days 北京站英雄帖一经发出&#xff0c;云原生的各路英雄豪杰纷纷响应。经典招式的升级亮相&#xff0c;最新技巧的惊喜面世&#xff0c;且看各路门派京城聚首&#xff0c;掀起一场云原生的武林论道。各大议…

深入解析Cloudflare五秒盾与爬虫绕过技巧

最近一个朋友发现一个比较有趣的网站&#xff0c;他说正常构造一个HTTP请求居然拿不到网站页面的信息&#xff0c;网站页面如下&#xff1a; 别看它只是一个普普通通的小说网站。随后我在本地环境验证了一下&#xff0c;果不其然得到了以下信息&#xff1a; 从上面反馈的信息…

Yakit: 集成化单兵安全能力平台使用教程·进阶篇

Yakit: 集成化单兵安全能力平台使用教程进阶篇 1.数据处理数据对比Codec2.插件仓库1.数据处理 数据对比 该功能主要提供一个可视化的差异比对工具,用于分析两次数据之间的区别。使用场景可能包括:枚举用户名时比较登录成功和失败时服务器端反馈结果的差异、使用 Web Fuzzer…

【css3实现华为充电】那些你没想到的CSS效果之华为充电效果(附源码下载)

【写在前面】今天是高考的第二天&#xff0c;在这里我也祝各位学子能够旗开得胜&#xff0c;进入自己理想的大学&#xff0c;借着今天这个吉日我就和大家介绍一下如何用css实现华为充电效果。 涉及知识点&#xff1a;CSS3特效&#xff0c;华为充电特效实现&#xff0c;CSS属性f…

部署DR模式 LVS负载均衡群集

部署DR模式 LVS负载均衡群集 一、LVS-DR数据包流向分析二、DR模型的特点三、DR模式 LVS负载均衡群集部署 一、LVS-DR数据包流向分析 &#xff08;1&#xff09;客户端发送请求到 Director Server&#xff08;负载均衡器&#xff09;&#xff0c;请求的数据报文&#xff08;源 …

SYSU程设c++(第十五周)

vector容器 1.要开vector库 2.vector<T> 是动态的连续数组&#xff0c;可以列表初始化 vector<int> ivec(10, 2); //创建10个值为2的元素 3.可以靠[ ]、at(int)、front、back、迭代器访问其中元素&#xff0c;其中at会自动检查下标越界&#xff0c;抛出异常 4.迭…

【资料分享】ESD防护设计-常见ESD保护电路图

ESD防护设计 ESD防护设计的目的是&#xff0c;当集成电路任意两个输入/输出引脚之间发生ESD事件时&#xff0c;集成电路内部的ESD防护系统能及时开启来泄放掉大量的瞬时电流/电压,使内部电路免遭破坏。此外&#xff0c;在集成电路正常工作时&#xff0c;即未发生ESD事件时&…

Jmeter Suite安装中influx一直处于pending状态

目录 【前言】 【背景说明】 【问题表现】 【排查思路】 简单重试 深入分析 直面本质 【小结】 【写在最后】 完整版文档下载方式&#xff1a; 【前言】 今天要和大家聊聊一个关于Jmeter Suite安装的问题——“influx一直处于pending状态”。 作为一名老测试&#…

文心一言 VS 讯飞星火 VS chatgpt (35)-- 算法导论5.3 5题

五、证明:在过程 PERMUTE-BY-SORTING的数组 P中&#xff0c;所有元素都唯一的概率至少是1-1/n。 文心一言&#xff1a; 证明&#xff1a; 在过程PERMUTE-BY-SORTING中&#xff0c;对于输入数组P中的每个元素&#xff0c;其出现次数为1&#xff0c;因此对于每个元素&#xff…

配置Linux服务器时间同步

这里说一下为什么要配置时间同步这个问题&#xff0c;为了避免各虚拟机的时间出现不一致的情况而引发集群故障。这里还要了解一下什么是系统时间&#xff0c;什么是硬件时间。   这里配置时间同步我选择的是Chrony。Chrony是网络时间协议&#xff08;NTP&#xff09;的通用实…

这11款协同软件大大提升你的工作效率

无论是产品经理还是开发工程师&#xff0c;他们都是各自领域的专家&#xff0c;所以他们可以成为设计过程的一部分。 协作设计软件为设计过程提供了沟通方式和工具&#xff0c;以便团队成员甚至客户能够交流、分享他们的意见和想法。 本文将盘点11个协作设计软件&#xff01;…

Jenkins DingTalk 钉钉通知插件

目录 前言 一、相关文档 二、组件版本 三、钉钉配置 四、Jenkins配置 1、安装钉钉插件DingTalk 2、在Jenkins用户管理中填写钉钉手机号 3、在Jenkins中配置钉钉 5、在流水线任务中编写pipeline 写在最后 完整版文档下载方式&#xff1a; 前言 今天要和大家聊聊一个…

docker安装golang

最近玩 docker 比较多&#xff0c;试试安装 golang 操作系统&#xff1a;Linux 第一步 先看一下镜像&#xff1a; docker images 看一下我们目前的镜像中&#xff0c;是不是有go 如果有&#xff0c;版本不合适等&#xff0c;可以考虑删除&#xff0c;重新安装&#xff0c;也…

【嵌入式烧录/刷写文件】-1.9-S19文件的地址对齐Address Alignment

案例背景(共5页精讲): 对一个Motorola S-record(S19/SREC/mot/SX)文件&#xff0c;进行地址对齐Address Alignment。 目录 1 为什么要进行“地址对齐Address Alignment” 1.1 “对齐长度”的选择 2 使用Vector HexView工具对S19文件进行“地址对齐Address Alignment” 2.1…

SpringBoot统一功能的处理

目录 SpringBoot拦截器 自定义拦截器 将自定义拦截器加入系统配置项 统一异常处理 创建异常处理类 实现业务方法 统一数据返回格式 为什么需要统一数据返回格式 统一数据返回格式的实现 返回String时的报错愿意及处理方案 SpringBoot拦截器 使用SpringBoot拦截器来实现…

Figma文件保存图片教程,4个详细步骤一网打尽!

在设计领域中&#xff0c;Figma 作为一款备受欢迎的在线协同设计软件&#xff0c;给设计师们带来了许多便利。然而&#xff0c;Figma 目前并不能直接保存图片。那么&#xff0c;有没有什么方法可以解决这个问题呢&#xff1f;答案是使用「即时设计」&#xff0c;这是一款国产的…

Git初始化提交项目代码并与远端建立连接

如题 闲来无事&#xff0c;炒个冷饭。。。 步骤 1. 先本地建仓库 方法一&#xff1a; 执行命令&#xff08;使用git bash或者类似工具&#xff0c;或者IDEA下terminal命令行&#xff09;&#xff1a; git init会在当前目录下创建一个新的空Git库。 方法二&#xff1a; 在…