C++中四种cast转换


C++中四种类型转换是:static_cast, dynamic_cast, const_cast, reinterpret_cast

1.const_cast

用于将const变量转为非const

2.static_cast

用于各种隐式转换,比如非const转const,void*转指针等, static_cast能用于多态向上转化,如果向下转能成功但是不安全,结果未知;
在这里插入图片描述

注:从图中可以看出来,派生类不仅有自己的方法和属性,同时它还包括从父类继承来的方法和属性。当我们从派生类向基类转化时,不管用传统的C语言还是C++转换方式都可以百分百转换成功,但是可怕的是向下转换,也就是由基类向派生类转换,当我们采用传统的C语言和C++转换时,就会出现意想不到的情况,因为转换后派生类自己的方法和属性丢失了,一旦我们去调用派生类的方法和属性那就糟糕了,这就是对类继承关系和内存分配理解不清晰导致的,好在C++增加了static_cast和dynamic_cast运用于继承关系类间的强制转换。

使用方法:

static_cast< new_type >(expression)

备注:new_type为目标数据类型,expression为原始数据类型变量或者表达式。

static_cast详解

static_cast相当于传统的C语言里的强制转换,该运算符把expression转换为new_type类型,用来强迫隐式转换如non-const对象转为const对象,编译时检查,用于非多态的转换,可以转换指针及其他,但没有运行时类型检查来保证转换的安全性。它主要有如下几种用法:

  1. 用于类层次结构中基类(父类)和派生类(子类)之间指针或引用的转换。
  2. 进行上行转换(把派生类的指针或引用转换成基类表示)是安全的;
  3. 进行下行转换(把基类指针或引用转换成派生类表示)时,由于没有动态类型检查,所以是不安全的。
  4. 用于基本数据类型之间的转换,如把int转换成char,把int转换成enum。
  5. 把空指针转换成目标类型的空指针。
  6. 把任何类型的表达式转换成void类型。

基本类型数据转换举例如下:

char a = 'a';
int b = static_cast<char>(a);//正确,将char型数据转换成int型数据

double *c = new double;
void *d = static_cast<void*>(c);//正确,将double指针转换成void指针

int e = 10;
const int f = static_cast<const int>(e);//正确,将int型数据转换成const int型数据

const int g = 20;
int *h = static_cast<int*>(&g);//编译错误,static_cast不能转换掉g的const属性

类上行和下行转换:

class Base
{};

class Derived : public Base
{}

Base* pB = new Base();
if(Derived* pD = static_cast<Derived*>(pB))
{}//下行转换是不安全的(坚决抵制这种方法)

Derived* pD = new Derived();
if(Base* pB = static_cast<Base*>(pD))
{}//上行转换是安全的

3.dynamic_cast

用于动态类型转换。只能用于含有虚函数的类,用于类层次间的向上和向下转化。只能转指针或引用。向下转化时,如果是非法的对于指针返回NULL,对于引用抛异常。要深入了解内部转换的原理。

向上转换:指的是子类向基类的转换

向下转换:指的是基类向子类的转换

它通过判断在执行到该语句的时候变量的运行时类型和要转换的类型是否相同来判断是否能够进行向下转换。

转换方式:

dynamic_cast< type >(e)*
 type必须是一个类类型且必须是一个有效的指针
dynamic_cast< type& >(e)
type必须是一个类类型且必须是一个左值
dynamic_cast< type&& >(e)
type必须是一个类类型且必须是一个右值

e的类型必须符合以下三个条件中的任何一个:
1、e的类型是目标类型type的公有派生类
2、e的类型是目标type的共有基类
3、e的类型就是目标type的类型。

如果一条dynamic_cast语句的转换目标是指针类型并且失败了,则结果为0。如果转换目标是引用类型并且失败了,则dynamic_cast运算符将抛出一个std::bad_cast异常(该异常定义在typeinfo标准库头文件中)。e也可以是一个空指针,结果是所需类型的空指针。

dynamic_cast主要用于类层次间的上行转换下行转换,还可以用于类之间的交叉转换(cross cast)。

在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。dynamic_cast是唯一无法由旧式语法执行的动作,也是唯一可能耗费重大运行成本的转型动作。

指针类型

举例,Base为包含至少一个虚函数的基类,Derived是Base的共有派生类,如果有一个指向Base的指针bp,我们可以在运行时将它转换成指向Derived的指针,代码如下:

if(Derived *dp = dynamic_cast<Derived *>(bp)){
  //使用dp指向的Derived对象  
}
else{
  //使用bp指向的Base对象  
}

值得注意的是,在上述代码中,if语句中定义了dp,这样做的好处是可以在一个操作中同时完成类型转换和条件检查两项任务。

引用类型

因为不存在所谓空引用,所以引用类型的dynamic_cast转换与指针类型不同,在引用转换失败时,会抛出std::bad_cast异常,该异常定义在头文件typeinfo中。

void f(const Base &b){
 try{
   const Derived &d = dynamic_cast<const Base &>(b);  
   //使用b引用的Derived对象
 }
 catch(std::bad_cast){
   //处理类型转换失败的情况
 }
}

4.reinterpret_cast

几乎什么都可以转,比如将int转指针,可能会出问题,尽量少用;

5.为什么不使用C的强制转换?

C的强制转换表面上看起来功能强大什么都能转,但是转化不够明确,不能进行错误检查,容易出错。

参考链接

1.说一说c++中四种cast转换
2.static_cast和dynamic_cast详解

  • 2
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++,有四种类型转换操作符,它们分别是static_cast、dynamic_cast、const_cast和reinterpret_cast。\[1\] 1. static_cast:用于基本类型之间的转换,例如将一个浮点数转换为整数,或者将一个整数转换为浮点数。它还可以用于类的上行和下行转换,即将派生类指针或引用转换为基类指针或引用,或者将基类指针或引用转换为派生类指针或引用。\[1\]\[2\] 2. dynamic_cast:用于在继承关系进行安全的向下转型。它可以将基类指针或引用转换为派生类指针或引用,但是只有在运行时能够确定转换是安全的情况下才会成功。如果转换不安全,dynamic_cast会返回空指针或引发std::bad_cast异常。\[1\] 3. const_cast:用于去除const属性,即将const修饰的对象转换为非const对象。它可以用于指针、引用和对象。\[1\] 4. reinterpret_cast:是最强大、最底层、最不安全的转换操作符。它可以将一个指针转换为一个整数,或者将一个整数转换为一个指针。它还可以进行不同类型指针之间的转换,但是这种转换是非常危险的,因为它不会进行类型检查。\[1\]\[3\] 这些转换操作符在C++提供了灵活的类型转换功能,但是在使用时需要谨慎,确保转换的安全性和正确性。 #### 引用[.reference_title] - *1* [C++四种cast转换](https://blog.csdn.net/Awesomewan/article/details/117112781)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [C++ 四种cast 详解](https://blog.csdn.net/m0_46210273/article/details/121147406)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值