C++进阶 类型转换

news2024/10/6 10:28:49

本文简介:介绍C++中类型转换的方式

类型转换

    • C语言中的类型转换
    • 为什么C++需要四种类型转换
    • C++强制类型转换
      • static_cast
      • reinterpret_cast
      • const_cast
      • dynamic_cast
    • RTTI(了解)
    • 总结

C语言中的类型转换

在C语言中,如果赋值运算符左右两侧类型不同,或者形参与实参类型不匹配,或者返回值类型与接收返回值类型不一致时,就需要发生类型转化,C语言中总共有两种形式的类型转换:隐式类型转换和显式类型转换。

  1. 隐式类型转化:编译器在编译阶段自动进行,能转就转,不能转就编译失败
  2. 显式类型转化:需要用户自己处理

隐式类型转换

   int i = 1;
   double d = i;

显式类型转换

    int*p = &i;
    int address = (int)p; 

而C++是C语言的超集 自然也能做到上面的两种类型转化

但是上面的两种类型转换有一种缺点

转换的可视性比较差,所有的转换形式都是以一种相同形式书写,难以跟踪错误的转换

为什么C++需要四种类型转换

C风格的转换格式很简单,但是有不少缺点的:

  1. 隐式类型转化有些情况下可能会出问题:比如数据精度丢失
  2. 显式类型转换将所有情况混合在一起,代码不够清晰

比如说下面这段代码

int end = 3;
size_t pos = 0;
while (1)
{ 
  if (end >= pos)
  {
    end--;
  }
}

上面的这段代码会进入死循环 因为在进行比较运算符运算的时候end发生了隐式类型转化
变成了size_t 类型 从此以后不会出现小于0的数了

因此C++提出了自己的类型转化风格,注意因为C++要兼容C语言,所以C++中还可以使用C语言的转化风格。

C++强制类型转换

标准C++为了加强类型转换的可视性,引入了四种命名的强制类型转换操作符:

static_cast、reinterpret_cast、const_cast、dynamic_cast

static_cast

static_cast用于非多态类型的转换(静态转换),编译器隐式执行的任何类型转换都可用
static_cast,但它不能用于两个不相关的类型进行转换

我们可以简单理解为是隐式类型转换 代码使用如下

int main()
{
 double d = 12.34;
 int a = static_cast<int>(d);
 cout<<a<<endl;
 return 0;
}

reinterpret_cast

reinterpret_cast操作符通常为操作数的位模式提供较低层次的重新解释,用于将一种类型转换为另一种不同的类型

我们可以简单理解为类型不相关的数据就可以使用 代码表示如下

int main()
{
 double d = 12.34;
 int a = static_cast<int>(d);
 cout << a << endl;
 // 这里使用static_cast会报错,应该使用reinterpret_cast
 //int *p = static_cast<int*>(a);
 int *p = reinterpret_cast<int*>(a);
 return 0;
}

const_cast

const_cast最常用的用途就是删除变量的const属性 方便赋值

我们可以写出以下代码

#include <iostream>    
using namespace std;    
    
    
int main()    
{    
  const int a = 2;    
  int* p = const_cast<int*>(&a);    
    
  *p = 3;    
    
  cout << a << endl;    
  cout << *p << endl;                                                                                                           
  return 0;    
} 

此时我们就可以通过p地址的解引用来改变该地址的值

那么大家可以猜猜看最后的结果是什么呢?

在这里插入图片描述
答案是2 3 是不是出乎大家的意料了

这是因为g++编译器认为 a是一个常量 我们不会去轻易修改它 所以说将a的值放在了寄存器当中 读取a的数据也会从寄存器中读

所以说尽管a这个变量的地址中的值被修改成3了 我们读取仍然会是2

那么有没有什么办法可以优化呢?

当然有 我们只需要让这个变量不出现在寄存器当中就可以了 我们可以使用volatile关键字修饰a变量 这样子a变量就不会出现在寄存器当中了 代码和演示结果如下

 volatile const int a = 2; 

在这里插入图片描述

dynamic_cast

我们在之前的内容中学习过 大部分的类型转换都是经过临时变量的 就比如说下面的代码

int a = 0;
double b = a;

这中间并不是直接将a赋值给b 而是会经历一个临时变量

在这里插入图片描述

这也就是为什么我们double类型的引用并不能直接引用int类型的数据 因为我们引用的实际上是一个临时变量 而临时变量具有常属性 必要要用const修饰才行 否则会产生一个权限扩大的问题

dynamic_cast用于将一个父类对象的指针/引用转换为子类对象的指针或引用(动态转换)

向上转型:子类对象指针/引用->父类指针/引用(不需要转换,赋值兼容规则)
向下转型:父类对象指针/引用->子类指针/引用(用dynamic_cast转型是安全的)

在C++中 子类对象的指针或者引用给父类指针或者引用的时候是不会经过类型转换的 因为这实际上就是一个切片 这是C++规则所允许的 (关于切片在我的继承博客那一章有讲解)

而向下转型 即父类对象指针/引用->子类指针/引用则是要经过类型转换的

首先我们要理解第一个点 父类的对象不管怎么样是绝对不被允许转化为子类对象的 只有指针和引用可以转

其实父类对象的指针有可能是指向父类的 也有可能是指向子类的

拿下面的两个类来举例说

class A
{
public:
	virtual void P()
	{
		;
	}
private:
	int _a = 1;
};

class B  : public A
{
private:
	int _b = 2;
};

父类是A 子类是B

void Test(A* pa)
{
	B* pb1 = dynamic_cast<B*>(pa);
	cout << pb1 << endl;
}

int main()
{
	A* pa = new A;
	B* pb = new B;
	Test(pa);
	Test(pb);
	return 0;
}

我们定义了一个父类指针 一个子类指针 并且将他们都传入到一个Test函数中去 子类的指针会被转化为父类的指针

但是我们需要知道的一点是 :父类的范围一定是小于等于子类的范围的

在这里插入图片描述

也就是说 父类的指针如果它本来是就是子类的 那么转化为子类指针之后正常的操作没有问题

但是如果说 父类指针本来就是指向父类的 那么转化为子类指针之后就有可能会出现越界问题

所以说为了解决指针越界问题 我们的dynamic_cast函数会先进行判断

如果该父类指针本来是指向子类的 那么可以成功转换 反之则返回一个空指针

运行结果如下

在这里插入图片描述

注意:

  1. dynamic_cast只能用于父类含有虚函数的类
  2. dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回0
  3. dynamic_cast相比于原来的强制类型转换是一种更加安全的转换方式

所以说dynamic_cast最大作用还是用来区分指针到底是指向父类还是指向子类

如果是指向父类的就返回空 如果是指向子类的就可以转

面试题

四种类型转换分别是什么
static_cast reinterpret_cast const_cast dynamic_cast

他们的应用场景分别是什么

  • static_cast 隐式类型转换
  • reinterpret_cast 强制类型转换
  • const_cast const修饰取消
  • dynamic_cast 父类指针/引用转子类指针/引用

RTTI(了解)

RTTI:Run-time Type identification的简称,即:运行时类型识别。

C++通过以下方式来支持RTTI:

  1. typeid运算符 获取变量类型
  2. dynamic_cast运算符 获取父类指针指向父类还是子类
  3. decltype 推导函数类型

总结

在这里插入图片描述

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

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

相关文章

数据同步后数据总条数对不上的问题解决

文章目录 [toc] 1.问题2.解决办法2.1&#xff09;设置合理的线程池参数2.2&#xff09;设置url连接参数2.3) 优化msql的系统参数2.4&#xff09;使用CountDownLatch减法计数器和数据插入的公共方法新开一个事务2.5&#xff09;sql批量注入器执行成功后&#xff0c;当前线程slee…

蛊卦-拨乱反正

目录 前言 卦辞 爻辞 总结 前言 题外话&#xff0c;今天占卜时&#xff0c;看错了&#xff0c;以为占到了蛊卦&#xff08;后续会对自己的占卦经历进行补充&#xff0c;不断完善这个易经学习的专栏&#xff09;&#xff0c;那顺便就学习一下蛊卦&#xff0c;蛊惑人心&#…

线程和进程同步互斥你真的掌握了吗?(同步互斥机制保姆级讲解与应用)

目录 同步互斥的概念 互斥锁 初始化互斥锁 销毁互斥锁 申请上锁 解锁 案例1&#xff1a;没有互斥锁 多任务的运行情况 案例2&#xff1a;有互斥锁 多任务的运行情况 死锁 读写锁 初始化读写锁 销毁读写锁 申请读锁 申请写锁 释放读写锁 案例&#xff1a;两个任务…

回归预测 | MATLAB实现GA-RBF遗传算法优化径向基函数神经网络多输入单输出回归预测(多指标,多图)

回归预测 | MATLAB实现GA-RBF遗传算法优化径向基函数神经网络多输入单输出回归预测&#xff08;多指标&#xff0c;多图&#xff09; 目录 回归预测 | MATLAB实现GA-RBF遗传算法优化径向基函数神经网络多输入单输出回归预测&#xff08;多指标&#xff0c;多图&#xff09;效果…

从0到1:通用后台管理系统 echarts图使用及其参数

这一章主要讲在系统概览模块中&#xff0c;所使用的echarts图及其参数 echarts是一个基于 JavaScript 的开源可视化图表库&#xff0c; 官网直通车 是在各种后台管理系统的开发中都常见的一种库&#xff0c;也是前端开发管理系统所必学的一种库 那么在项目中主要是使用了饼…

【前端】React快速入门+Redux状态管理

本文旨在记录react的基础内容&#xff0c;帮助有需要的同学快速上手,需要进一步了解描述更加稳妥和全面的信息&#xff0c;请查阅官方文档 官方文档点击这里进行跳转 React快速入门 先导 react框架 vue,react,angular这几种主流前端框架使用频率较高…本质还是js库。 React…

1.Jetson Orin Nano Developer Kit系统刷机

本教程有3种方法刷机&#xff0c;根据需要自己选择适合自己的方案。 一:使用>32G的SD卡安装开发套件; 二:在Ubuntu18.04下通过SDK Manager软件在线安装系统. 三:在Ubuntu18.04下通过脚本方式安装系统. 注意&#xff1a;Ubuntu的账号不能为一些常见的包名如&#xff1a;p…

【学习日记】【FreeRTOS】FreeRTOS 移植到 STM32F103C8

前言 本文基于野火 FreeRTOS 教程&#xff0c;内容是关于 FreeRTOS 官方代码的移植的注意事项&#xff0c;并将野火例程中 STM32F103RC 代码移植到 STM32F103C8。 一、FreeRTOS V9.0.0 源码的获取 两个下载链接&#xff1a; 官 网 代码托管 二、源码文件夹内容简介 Source…

Docker(一)-安装、架构、业务开发常用命令、Dockerile、镜像卷、镜像仓库

基于业务开发使用Docker Docker是一个开源的容器引擎&#xff0c;它有助于更快地交付应用。Docker可将应用程序和基础设施层隔离&#xff0c;并且能将基础设施当作程序一样进行管理。使用 Docker可更快地打包、测试以及部署应用程序&#xff0c;并可以缩短从编写到部署运行代码…

为什么需要单元测试?

为什么需要单元测试&#xff1f; 从产品角度而言&#xff0c;常规的功能测试、系统测试都是站在产品局部或全局功能进行测试&#xff0c;能够很好地与用户的需要相结合&#xff0c;但是缺乏了对产品研发细节&#xff08;特别是代码细节的理解&#xff09;。 从测试人员角度而言…

Springboot 实践(10)spring cloud 与consul配置运用之服务的注册与发现

前文讲解&#xff0c;完成了springboot、spring security、Oauth2.0的继承&#xff0c;实现了对系统资源的安全授权、允许获得授权的用户访问&#xff0c;也就是实现了单一系统的全部技术开发内容。 Springboot是微服务框架&#xff0c;单一系统只能完成指定系统的功能&#xf…

【简单认识Docker网络管理】

文章目录 一、Docker 网络实现原理二、Docker 的网络模式1.四种网络模式2.各网络模式详解&#xff08;1&#xff09;Host模式&#xff08;2&#xff09;Container模式&#xff08;3&#xff09;None模式&#xff08;4&#xff09;Bridge模式 3.指定容器网络模式4.自定义网络模式…

web文件上传

文件上传指的是&#xff0c;将本地的图片、视频、音频上传到服务器&#xff0c;提供给其他用户浏览和下载的过程 前端需求 想要进行文件上传对于web前端来说有三个重要要素 1.<input type"file" name"image"> 提供这样的file文件上传格式 2. metho…

【Unity】自带的录屏插件Recorder

目录 Recorder简介Recorder导入Recorder使用 Recorder简介 Recorder是Unity官方的录屏插件&#xff0c;可以直接录制Game窗口&#xff0c;还可以录制不同相机的视图。不仅可以直接生成视频、帧动画图、还可以制作gif和animation。 Recorder导入 菜单栏Windows→Package Mana…

VINS-Mono中的边缘化与滑窗 (4)——VINS边缘化为何是局部变量边缘化?

文章目录 0.前言1.系统构建1.1.仿真模型1.2.第一次滑窗优化1.3.第二次全局优化 2.边缘化时不同的舒尔补方式2.1.边缘化时舒尔补的意义2.2.不同的边缘化方式 3.边缘化时不同的舒尔补方式实验验证3.1.全局schur的操作方式3.2.VIO或VINS中局部边缘化的方式3.3.两种方式和全局优化方…

【Linux】文件的描述符和重定向

文件的描述符和重定向 C语言的文件读写操作代码 open系统打开文件方法系统读写文件操作文件描述符文件重定向怎么理解文件缓冲区 C语言的文件读写操作 文件写入 fputs int fputs(const char *s, FILE *stream); s&#xff1a;要写入的字符串 stream&#xff1a;要写入对应的目标…

halcon库文件封装操作方法介绍

头文件写法参考 getarea.h 导出函数设定&#xff1a; extern “C” __declspec(dllexport) void _stdcall cpp文件写法参看 3.vs 库文件配置介绍 头文件及包含的库文件目录设定 &#xff08;1&#xff09;头文件设定E:\halcon\env\opencv\include\opencv2;E:\halcon\env\ope…

接口限流注解 RateLimiter (现成直接用)

1. 限流注解 import com.ruoyi.common.constant.Constants; import com.ruoyi.common.enums.LimitType;import java.lang.annotation.*;/*** 限流注解* */ Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface RateLimiter {/*** 限流…

DSO 系列文章(3)——DSO后端正规方程构造与Schur消元

文章目录 DSO代码注释&#xff1a;https://github.com/Cc19245/DSO-CC_Comments

javascript期末作业【三维房屋设计】 【文末源码+文档下载】

1、引入three.js库 官网下载three.js 库 放置目录并引用 引入js文件: 设置场景&#xff08;scene&#xff09; &#xff08;1&#xff09;创建场景对象 &#xff08;2&#xff09;设置透明相机 1,透明相机的优点 透明相机机制更符合于人的视角,在场景预览和游戏场景多有使用…