C++进阶 类型转换

简介: C++进阶 类型转换

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


tatic_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地址的解引用来改变该地址的值

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

a319f12c837c4f5b844a057830c3f3cc.png

答案是2 3 是不是出乎大家的意料了

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

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


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

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

volatile const int a = 2;

8da4228b32914fa799b4f7ad981d2ab1.png


dynamic_cast

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

int a = 0;
double b = a;

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

f82d74784bbf44ebbbf08d2b8cd6d639.png

这也就是为什么我们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函数中去 子类的指针会被转化为父类的指针

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

3d14d67aff554e418e1418093540e151.png

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

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

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

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

运行结果如下

6bebdb76efca40468d90e71dd6844bfc.png

注意:

  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 推导函数类型


总结

0ae300dc03124cf0995fe955415dfe9a.png

相关文章
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
110 5
|
7月前
|
编译器 C++
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
61 1
|
3月前
|
编译器 C语言 C++
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
33 1
|
3月前
|
存储 编译器 数据安全/隐私保护
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解2
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
46 3
|
3月前
|
编译器 C++
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解1
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
58 3
|
7月前
|
存储 编译器 C++
C++进阶之路:何为拷贝构造函数,深入理解浅拷贝与深拷贝(类与对象_中篇)
C++进阶之路:何为拷贝构造函数,深入理解浅拷贝与深拷贝(类与对象_中篇)
57 0
|
3月前
|
C++
C++入门4——类与对象3-2(构造函数的类型转换和友元详解)
C++入门4——类与对象3-2(构造函数的类型转换和友元详解)
30 0
|
7月前
|
安全 算法 C语言
【C++进阶】深入STL之string:掌握高效字符串处理的关键
【C++进阶】深入STL之string:掌握高效字符串处理的关键
73 1
【C++进阶】深入STL之string:掌握高效字符串处理的关键
|
7月前
|
编译器 C++
C++模板进阶
C++模板进阶
29 1
|
7月前
|
存储 算法 程序员
【C++进阶】深入STL之 栈与队列:数据结构探索之旅
【C++进阶】深入STL之 栈与队列:数据结构探索之旅
65 4