网站建设公司 保证完成 /百度网页版入口链接
目录
前言
C语言中的类型转换
C++强制类型转换
static_cast(static静止的)
reinterpret_cast(reinterpret重新解释)
const_cast(const常量)
总结
dynamic_cast(dynamic动态)
RTTI
常见面试题
前言
C++继承了不少C语言的知识,因为C++最开始就是从C语言出来的,于是也就导致了C++将C语言的,好的与不好的,都继承下来了。
有一个东西就是从C语言沿袭过来的,但是其又是不够好的 —— 隐式类型的转换。
C语言中的类型转换
- 隐式类型转化:编译器在编译阶段自动进行,能转就转,不能转就编译失败。
- 显式类型转化:需要用户自己处理。
void Test ()
{int i = 1;// 隐式类型转换double d = i;printf("%d, %.2f\n" , i, d);int* p = &i;// 显示的强制类型转换int address = (int) p;printf("%x, %d\n" , p, address);
}
#问:什么情况下允许隐式类型转换?什么情况下允许强制类型转换?
(C++继承的C语言的 —— C语言那就当然是针对于内置类型啦)
C语言是意义相近的类型允许隐式类型的准换,如:整形家族和浮点数家族。
- 整形之间转换是:小的可以转大的(提升),大的也可以转小的(截断)。
- 整形和浮点之间转换是:通过补位的方式,整形转浮点补浮点就行因为精度,浮点转整形丢掉精度。
可以说:它们都是用来表示数据的大小,只是空间的大小不一样,表示的范围不一样。还有就是浮点数的存储机制也不一样,其还能表示小数点后的精度。(本质:意义相近)
C语言中对于显示强制类型转换是意义不相近的类型,值转换后有意义。
Note:
整体来说,隐式类型还好,但是显示类型是有巨大的坑的。
在日常中最简单的insert函数实现(数据移动)中,就有问题。
void Insert(size_t pos, char ch) {size_t _size = 5;// ……int end = _size - 1;while(end >= pos){_str[end + 1] = _str[_end];--end;} }Insert(3, 'A'); // 没有问题 Insert(0, 'A'); // 有问题
在其中,第二个Insert就存在问题,因为在判断的时候就会发生隐式类型的转换。
在这个判断的时候应该就是end为-1,pos为0然后,退出while循环,但是此处它会悄悄地进入执行,因为在一个操作符的两端的操作数也会发生隐式类型的转换。这个时候会悄悄的产生一个变量将end进行了提升(从无符号提升成有符号),然后就会导致数据的越界。
所以,C语言留下的这个东西也是一个坑。并且没有办法因为为了兼容前面的内容,所以没有办法移除隐式转换,所以我们需要注意隐式转换的坑。
于是,便有了C++的规范化。
#问:为什么C++需要四种类型转换?
- 隐式类型转化有些情况下可能会出问题:比如数据精度丢失。
- 显式类型转换将所有情况混合在一起,代码不够清晰。
Note:因为C++要兼容C语言,所以C++中还可以使用C语言的转化风格。
C++强制类型转换
标准C++为了加强类型转换的可视性,引入了四种命名的强制类型转换操作符:
- static_cast
- reinterpret_cast
- const_cast
- dynamic_cast
这个时候C语言的隐式类型转换的一套还是可以使用的,只不过一般编译器会报警告,不影响运送,但是会告诉你,如:浮点数转换为整形精度可能会丢失。
static_cast(static静止的)
static_cast用于非多态类型的转换(静态转换),编译器隐式执行的任何类型转换都可用static_cast,但它不能用于两个不相关的类型进行转换 - 意义相近的类型。
int main()
{double d = 12.34;int a = static_cast<int>(d);std::cout << a << std::endl;return 0;
}
不是相近的类型不可以运用其来转换。
int main()
{int* p = &a;// 不支持的int address = static_cast<int>(p);return 0;
}
其是会报错为,如:"static_cast":无法从"int*"转换为"int",的错误(无效的)。
reinterpret_cast(reinterpret重新解释)
reinterpret_cast操作符通常为操作数的位模式提供较低层次的重新解释,用于将一种类型转换为另一种不同的类型 - 意义不相关的类型。
int main()
{double d = 12.34;int a = static_cast<int>(d);int* p = &a;// 不支持的//int address = static_cast<int>(p);int address = reinterpret_cast<int>(p);return 0;
}
对于const修饰的变量,不能转化,因为C++的规范会进行检查。
int main()
{const int a = 2;// 不支持int* p = reinterpret_cast<int*>(&a)*p = 3;return 0;
}
补充:
甚至有一些地方还可以支持一些比较bug的转化。因为其相当于重新解释了,转换成完全另外一个类型。如:其实是一个指针,转换成为了一个数据大小,或者是一个数据大小转换成为了指针。
const_cast(const常量)
const_cast最常用的用途就是删除变量的const属性,方便赋值。
int main()
{const int a = 2;int* p = const_cast<int*>(&a)*p = 3;return 0;
}
一个奇葩的操作。
int main()
{const int a = 2;int* p = const_cast<int*>(&a);*p = 3;cout << a << endl;cout << *p << endl;return 0;
}
#问:上面的奇葩的操作的输出结果是什么?
简单的来看是不是以为是:3、3?那就错了。
p是a的地址,那*p就是a,这么想上面没错(const对象虽然不能改,但是在C++里面const修饰的变量叫做常变量,是不能被直接的修改,但是可以被间接接的修改)。
修改const变量:
就像上述,就是改const的方法。这种方式很bug,我们取到const修饰的变量的地址,然后通过地址间接的强制去改。本质的原因就是C++的const修饰的变量,并没有存储到常量区当中去,const修饰的变量与其他的变量一样,都存在栈上的,所以使用指针的方式是可以被修改的。
其实运行结果是:2、3!
而且我回很奇特的发现,我们对于数据进行监视数据是:3、3,然而实际打印出的数据却是2,3。
这个的原因是由来于编译器的优化,编译器对于const类型的变量是有优化的。比如说有一些编译器的优化是会将数据放在寄存器的,因为对于编译器来说,这个const修饰的对象,是只会读而并不会进行修改的,于是为了提高运行的效率,于是直接将数存储到了寄存器当中,你要就取。所以虽然内存中的数据被进行了更改,但是在寄存器看来就是没变。
有一些编译器的处理方式,不是放到寄存器,而是直接搞死,如同一个宏。不取直接就给初始化时的值。
所以也就是为什么 const_cast 也可以作为毫不相关的转换(重新解释的转换)。 const_cast 单独形成一类,也就是为了告诉我们这个地方很危险。还有一个方法可以解决这个问题:volatile(关键字),就是为了告诉编译器这个地方不要进行优化,直接去内存中取这个值。
总结
- 兼容C隐式类型转换和强制类型转换。
- 期望我们需要使用C语言的,期望我们用规范的C++显示的强制类型转换。
- static_cast(隐式类型转换)、reinterpret_cast、const_cast(强制类型转换)。
dynamic_cast(dynamic动态)
是C语言没有的,是C++自己增加的,适用于向下转化。dynamic_cast用于将一个父类对象的指针 / 引用转换为子类对象的指针或引用(动态转换)。
- 向上转型:子类对象指针/引用->父类指针/引用(不需要转换,赋值兼容规则) —— 天然支持
严格的来说:向上转型是不属于隐式类型转换,也不属于显示类型转换。它可以说是C++的一个特例,其不需要进行转换,它是赋值兼容的。
- 向下转型:父类对象指针/引用->子类指针/引用(用dynamic_cast转型是安全的)
- dynamic_cast只能用于父类含有虚函数的类 —— 否者编译直接报错。
- dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回0。
class A
{
public :virtual void f(){}
};class B : public A
{};int main ()
{B bb;A aa = bb;A& ra = bb;return 0;
}
dynamic_cast是适用于向下转型,父转子(用dynamic_cast转型是安全的)。因为按道理父类对象是无论如何都不能转子类的,就是使用dynamic_cast也是不行的。
class A
{
public :virtual void f(){}
};class B : public A
{};int main ()
{A aa;// 父类对象无论如何都是不允许转换成子类对象的// B bb = dynamic_cast<B>(aa);// B bb = (B)aa;return 0;
}
对象是怎么样都不允许转的,但是指针与引用是要允许转的。因为涉及到一个指针,是有可能指向父类,也有可能指向子类。而如果是父类指针指向是父类的对象,那么就会导致出现越界问题。而如果是父类指针指向是子类的对象,那就没有问题。于是C++提供了一个dynamic_cast,让我们的行为是安全的,即:
- 如果父类指针指向的是子类的对象,那么可以转换,转换表达式返回正确的地址。
- 如果父类指针指向的是父类的对象,那么不能转换,转换表达式返回nullptr。
class A
{
public :virtual void f(){}
};class B : public A
{};int main ()
{// 如果pa是指向子类,那么可以转换,转换表达式返回正确的地址// 如果pa是指向父类,那么不能转换,转换表达式返回nullptrB bb;A* pa = bb;B* pb = dynamic_cast<B*>(pa); // 安全的//B* pb = (B*)pa; // 不安全if (pb){cout << "转换成功" << endl;pb->_a++;pb->_b++;cout << pb->_a << ":" << pb->_b << endl;}else{cout << "转换失败" << endl;pa->_a++;cout << pa->_a << endl;}
}
总结:
其中,向上转型就是所说的切割/切片,是语法天然支持的,不需要进行转换,而向下转型是语法不支持的,需要进行强制类型转换。
#include <iostream>class A1
{
public:virtual void f(){}
public:int _a1 = 0;
};class A2
{
public:virtual void f(){}
public:int _a2 = 0;
};class B : public A1, public A2
{
public:int _b = 1;
};int main()
{B bb;A1* ptr1 = &bb;A2* ptr2 = &bb;std::cout << ptr1 << std::endl;std::cout << ptr2 << std::endl << std::endl;B* pb1 = (B*)ptr1;B* pb2 = (B*)ptr2; // C语言的强制类型转换是回得到开头的std::cout << pb1 << std::endl;std::cout << pb2 << std::endl << std::endl;B* pb3 = dynamic_cast<B*>(ptr1);B* pb4 = dynamic_cast<B*>(ptr2);std::cout << pb3 << std::endl;std::cout << pb4 << std::endl << std::endl;return 0;
}
Note:强制类型转换关闭或挂起了正常的类型检查,每次使用强制类型转换前,程序员应该仔细考虑是否还有其他不同的方法达到同一目的,如果非强制类型转换不可,则应限制强制转换值的作用域,以减少发生错误的机会。强烈建议:避免使用强制类型转换。
RTTI
C++通过以下方式来支持RTTI:
- typeid运算符 —— 拿到变量的类型的字符串。
- dynamic_cast运算符 —— 父类的指针指向父类对象,还是子类对象,只能用于父类含有虚函数的类(多态)。
- decltype —— 推导一个对象的类型,这个类型可以用来定义另一个对象。
常见面试题
#问:C++中的4种类型转换分别是:____ 、____ 、____ 、____。
static_cast、reinterpret_cast、const_cast和dynamic_cast。
#问:说说4种类型转换的应用场景。
- static_cast用于相近类型的类型之间的转换,编译器隐式执行的任何类型转换都可用static_cast。
- reinterpret_cast用于两个不相关类型之间的转换。
- const_cast用于删除变量的const属性,方便赋值。
- dynamic_cast用于安全的将父类的指针(或引用)转换成子类的指针(或引用)。
相关文章:

【C++】-- 类型转换
目录 前言 C语言中的类型转换 C强制类型转换 static_cast(static静止的) reinterpret_cast(reinterpret重新解释) const_cast(const常量) 总结 dynamic_cast(dynamic动态) …...

汇编基础语法和指令总结+案例(用32位汇编实现插入排序)
目录 前提知识 案例 c的插入排序 32位汇编代码 代码分析 效果展示 前提知识 常用指令add指令 sub指令 mul乘法指令 div除法指令 inc(自增)(即) dec(自减)(即--) cmp…...

C++多线程--线程安全的单例模式
0 引言 由于最近事情比较多,所以很久没有更新相应的专栏了。目前事情基本告一段落,重新恢复相应专栏的更新。 本文主要讲解在C++并发编程中如何实现线程安全的单例模式。本文主要由如下几部分构成 臭名昭著的double-check单例实现四种线程安全的单例模式单例模式使用中所带…...

(Android-RTC-9)PeerConnectionFactory
开篇前瞎扯。很久没发技术文章了,此文一直放着草稿箱没有完成,感觉自己在家庭和工作中找到了拖延的借口,开始慢慢变得懒惰了,那是万万不行的。恰逢2023开年ChatGPT的爆火,更让我这些普通程序员危机感瞬间飙升ÿ…...

Vector - CAPL - 定时器函数和使用
定时器在C语言中的使用我想学习过C编程的都不会陌生,它能够提供延时,完成等待一定的时间;它也可以实现多线程的操作,并行实行某些软件功能。那在CAPL中,定时器又能做哪些工作呢?又是怎么使用的呢࿱…...

【嵌入式C】常见问题
1、goto的使用场景有哪些?并讨论其局限? (1)常用来跳出死循坏; (2)在linux开发中,常用于打印错误; (3)goto在某些使用场合会破坏程序的栈逻辑&…...
[神经网络]Transfomer架构
一、概述 Transfomer架构与传统CNN和RNN最大的区别在于其仅依赖自注意力机制,而没有卷积/循环操作。其相较于RNN,不需要进行时序运算,可以更好的进行并行;相较于CNN,其一次可以关注全图而不局限于感受野尺寸。 二、模…...

C++之多态 虚函数表
多态 多态是在不同继承关系的类对象,去调用同一函数,产生了不同的行为。 需要区分一下:1、菱形虚拟继承,是在继承方式前面加上virtual; class Person {}; class Student : virtual public Person {}; class Teacher…...

AI_Papers周刊:第四期
2023.02.28—2023.03.05 Top Papers Subjects: cs.CL 1.Language Is Not All You Need: Aligning Perception with Language Models 标题:KOSMOS-1:语言不是你所需要的全部:将感知与语言模型相结合 作者:Shaohan Huang, Li …...

A Simple Framework for Contrastive Learning of Visual Representations阅读笔记
论文地址:https://arxiv.org/pdf/2002.05709.pdf 目前流行的无监督学范式。通过训练,使模型拥有比较的能力。即,模型能够区别两个数据(instance)是否是相同的。这在 深度聚类 领域受到广泛的关注。(在有监…...

mac安装开发工具:clipy、iterm2、go、brew、mysql、redis、wget等
wget brew install wget clipy Releases Clipy/Clipy GitHub 环境变量 ~下有三个文件 .zshrc .zprofile .bash_profile > cat .zshrc export PATH$PATH:/usr/local/mysql/bin> cat .zprofile eval "$(/opt/homebrew/bin/brew shellenv)"> cat .bas…...

DJ1-1 计算机网络和因特网
目录 一、计算机网络 二、Interent 1. Internet 的介绍 2. Internet 的具体构成 3. Internet 提供的服务 4. Internet 的通信控制 一、计算机网络 定义:是指两台以上具有独立操作系统的计算机通过某些介质连接成的相互共享软硬件资源的集合体。 计算机网络向…...

[1.3.3]计算机系统概述——系统调用
文章目录第一章 计算机系统概述系统调用(一)什么是系统调用,有何作用(二)系统调用与库函数的区别(三)小例子:为什么系统调用是必须的(四)什么功能要用到系统调…...

【Java开发】JUC进阶 03:读写锁、阻塞队列、同步队列
1 读写锁(ReadWriteLock)📌 要点实现类:ReentrantReadWirteLock通过读写锁实现更细粒度的控制,当然通过Synchronized和Lock锁也能达到目的,不过他们会在写入和读取操作都给加锁,影响性能&#x…...

Fragment中获取Activity的一点点建议
平时的Android开发中,我们经常要在Fragment中去获取当前的Activity实例,刚开始的时候可能使用使用Fragment提供的getActivity方法来获取,但是这个方法可能返回null,为了让程序可以正常运行,项目中就出现大量下面这样的…...

Java Math类
Java Math 类是 Java 标准库中提供的一个数学计算类,它提供了很多数学函数,如三角函数、指数函数、对数函数等。在实际工作中,Java Math 类常常被用于处理数学计算问题,例如计算复杂的数学公式、实现数学算法等。本文将详细介绍 J…...

Javascript -- 加载时间线 正则表达式
js加载时间线 1、创建Document对象,开始解析web页面,解析html元素和他们的文本内容后添加Element对象和Text节点到文档中。这个阶段的document.readyState ‘loading’ 2、遇到link外部css,创建线程加载,并继续解析文档 3、遇到…...

gdb/git的基本使用
热爱编程的你,一定经常徘徊在写bug和改bug之间,调试器也一定是你随影而行的伙伴,离开了它你应该会寝食难安吧! 目录 gdb的使用 断点操作 运行调试 观察数据 Git的使用 仓库的创建和拉取 .gitignore “三板斧” 常用指令 gd…...

信息安全与数学基础-笔记-④二次同余方程
知识目录二次同余方程的解欧拉判别式Legendre (勒让德符号)二次同余方程的解 什么是二次同余方程的解 注意这里二次同余方程和一次同余方程是不一样的 在x2x^2x2 三 a (mod m) 方程中举例 ↓ 解即剩余类,因为是模m,所以我们在 [ 0, m-1 ]中逐个代入看是…...

Luogu P4447 [AHOI2018初中组]分组
题目链接:传送门 将nnn个可重复的整数分为mmm组,每组中的数必须连续且不重复,使人数最少的组人数最多。 两个最值肯定第一想到二分,每次二分出一个值,判断在这个值为答案的前提下能否完成分组。 在思考判别函数时发现…...

手把手创建flask项目
Flask 框架流程 什么是Flask: Flask诞生于2010年, 使用python语言基于Werkzeug工具箱编写的轻量级Web开发框架 Flask本身相当于一个内核, 其他几乎所有的功能都要用到扩展(邮件:Flask-Mail, 用户认证:Flask-Login, 数据库:Flask-SQLAlchemy). Flask的核心在于Werkz…...

SpringCloud-4_Eureka服务注册与发现
Eureka作为一个老牌经典的服务注册&发现技术,其设计和理念,也在影响后面的组件。目前主流的服务注册&发现的组件是Nacos当前项目架构问题分析-引出Eureka问题分析:1.在企业级项目中,服务消费访问请求会存在高并发2.如果只…...

【react全家桶】生命周期
文章目录04 【生命周期】1.简介2.初始化阶段2.1 constructor2.2 componentWillMount(即将废弃)2.3 static getDerivedStateFromProps(新钩子)2.4 render2.5 componentDidMount2.6 初始化阶段总结3.更新阶段3.1 componentWillRecei…...

虚拟机安装Windows 10
虚拟机安装Windows 10 镜像下载 方法一:下载我制作好的镜像文件->百度网盘链接 提取码:Chen 方法二:自己做一个 进入微软官网链接 下载"MediaCreationTool20H2" 运行该工具 点击下一步选择路径,等他下载好就欧克了…...

【CMU15-445数据库】bustub Project #2:B+ Tree(下)
Project 2 最后一篇,讲解 B 树并发控制的实现。说实话一开始博主以为这块内容不会很难(毕竟有 Project 1 一把大锁摆烂秒过的历史x),但实现起来才发现不用一把大锁真的极其痛苦,折腾了一周多才弄完。 本文分基础版算法…...

leetcode 困难 —— 外星文字典(拓扑排序)
题目: 现有一种使用英语字母的外星文语言,这门语言的字母顺序与英语顺序不同。 给定一个字符串列表 words ,作为这门语言的词典,words 中的字符串已经 按这门新语言的字母顺序进行了排序 。 请你根据该词典还原出此语言中已知的字…...

ubuntu server 18.04使用tensorflow进行ddqn训练全过程
0. 前言 需要使用ddqn完成某项任务,为了快速训练,使用带有GPU的服务器进行训练。记录下整个过程,以及遇到的坑。 1. 选择模板代码 参考代码来源 GitHub 该代码最后一次更新是Mar 24, 2020。 环境配置: python3.8 运行安装脚本…...

2023年全国最新二级建造师精选真题及答案14
百分百题库提供二级建造师考试试题、二建考试预测题、二级建造师考试真题、二建证考试题库等,提供在线做题刷题,在线模拟考试,助你考试轻松过关。 二、多选题 61.已经取得下列资质的设计单位,可以直接申请相应类别施工总承包一级…...

mysql一条语句的写入原理
mysql写入原理 我们知道在mysql数据库最核心的大脑就是执行引擎; 其中的默认引擎Innodb在可靠执行和性能中做出来平衡; innodb支持在事务控制、读写效率,多用户并发,索引搜索方面都表现不俗; innodb如何进行数据写入…...

嵌入式Linux内核代码风格(二)
第九章:你已经把事情弄糟了 这没什么,我们都是这样。可能你的使用了很长时间Unix的朋友已经告诉你“GNU emacs”能 自动帮你格式化C源代码,而且你也注意到了,确实是这样,不过它所使用的默认值和我们 想要的相去甚远&a…...