【C++11】四种强制类型转换 与 C语言的类型转换

C语言的类型转换

首先我们分析C语言中常见的类型转换:

C语音共有两种类型转换:

  • 隐式类型转化:编译器在编译阶段自动进行,可以转化便转化,不能转就编译失败
  • 显式类型转化:用户自行处理

观察下面的代码:

void Test2()
{
    // --------------- 隐式类型转换: (用于意义相近的类型) ---------------
    // 自动类型转换
    int a = 65;
    double b = a; // 自动将整型a转换为浮点型b
    printf("a: %d, b: %f\n", a, b);


    //--------------- 显式类型转换 ---------------
    // 强制类型转换
    double c = 3.14;
    int d = (int)c; // 将浮点型c强制转换为整型d
    printf("c: %f, d: %d\n", c, d);

    // sizeof运算符转换
    unsigned long e = sizeof(int); // 将sizeof(int)的结果转换为unsigned long类型
    printf("Size of int: %lu bytes\n", e);

    // 高风险的类型转换
    int* ptr = NULL;
    double* ptr_double = (double*)ptr; // 将int指针强制转换为double指针
    printf("ptr: %p, ptr_double: %p\n", ptr, ptr_double);
}

观察上面的代码可以看到 c语言 类型转化存在一定的缺陷

  1. 转换的可视性比较差,所有的转换形式都是以一种相同形式书写,难以跟踪错误的转换
  2. 隐式类型转化有些情况下可能会出问题:比如数据精度丢失
  3. 显式类型转换将所有情况混合在一起,代码不够清晰

C++强制类型转换

C++引入了四种命名的强制类型转换操作符:static_cast、reinterpret_cast、const_cast、dynamic_cast

四种转换的语法相同(以static_cast举例),均为:

dynamic_cast<new_type>(expression)

其中,new_type是你要转换为的新类型,expression是你要转换的指针或引用


static_cast

static_cast 允许在编译时执行兼容类型之间的类型转换。

  • static_cast可以用于在内置类型之间进行转换,例如将 int转换为double,也可以在用户定义的类型之间进行转换,例如将指向基类的指针转换为指向派生类的指针

  • 需要注意的是,static_cast仅执行简单的转换。它可以执行不涉及底层值变化的转换,例如将int转换为long,但它不能执行涉及精度或信息丢失的转换,例如通过截断小数部分将float转换为int。

void Test_static_cast()
{
    double d = 11.45;
    int a = static_cast<int>(d);
    cout << a << endl; //11
}

该代码中发生了截断


reinterpret_cast

  • reinterpret_cast操作符通常为操作数的位模式提供较低层次的重新解释,用于将一种类型转换为另一种不同的类型
void Test_reinterpret_cast()
{
    double d = 11.4;
    int a = static_cast<int>(d);
    cout << a << endl;

    // static_cast只能执行简单的转换,即不能执行指针类型之间的转换
    // int* p = static_cast<int*>(a); //err 类型转换无效
    int* p = reinterpret_cast<int*>(a);
}

在此代码中,reinterpret_cast 用于将一个 int 值转换为一个 int* 指针


const_cast

const_cast 用于将常量类型转换为非常量类型,或者将非常量类型转换为常量类型。通常,const_cast 用于删除或添加常量性,以便在函数调用或表达式中传递常量数据

void Test_const_cast()
{
    const int a = 1;
    int* p = const_cast<int*>(&a); //删除变量的const属性,方便赋值
    *p = 4;

    cout << a << endl; //1
}

dynamic_cast

dynamic_cast 用于在运行时执行安全的 向下转型(Downcasting) :即将一个基类的指针 / 引用 转换 为一个派生类的指针 / 引用

同时使用 dynamic_cast 时需要注意:

  1. dynamic_cast 只能用于父类含有虚函数的类
  2. dynamic_cast先检查是否能转换成功,能成功则转换,不能则返回0。
class A
{
public:
    virtual void f(){} //虚函数f
};

class B:public A // B类继承A类
{};

void func(A* pa)
{
    //dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回0
    B* pb1 = static_cast<B*>(pa);
    B* pb2 = dynamic_cast<B*>(pa);

    cout << "pb1:" << pb1 << endl;
    cout << "pb2:" << pb2 << endl;
}

void Test_dynamic_cast()
{
    A a;    B b;
    func(&a);   func(&b);
}

对于上述代码的分析:

由于static_cast不进行运行时类型检查,它可以将任何指针类型转换为任何其他指针类型,即使它们没有继承关系。在这个例子中,static_cast<B*>(pa)将A类型的指针pa直接转换为B类型的指针,不进行任何安全检查。因此,如果pa指向一个不是B类型的对象,则会导致未定义的行为。

相比之下,dynamic_cast会在运行时进行类型检查,确保转换是安全的。如果pa指向的对象是B类型或B类型的子类,dynamic_cast<B*>(pa)将返回一个指向B类型的指针。如果pa指向的对象不是B类型或B类型的子类,则dynamic_cast将返回一个空指针。

因此,在这个例子中,static_cast和dynamic_cast的结果是不同的

static_cast<B*>(pa)可能会导致未定义的行为,而dynamic_cast<B*>(pa)则会在运行时进行类型检查,确保转换是安全的 。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C语言中,有四种强制类型转换操作符,分别是const_cast、dynamic_cast、reinterpret_cast和static_cast。这些转换类型适用于不同的目的。const_cast主要用于强制取消对象的常量性,它是唯一能够做到这一点的操作符。dynamic_cast主要用于在运行时进行安全的类型转换,它可以在继承关系中进行向下转型,并且会进行安全检查以确保类型转换的有效性。reinterpret_cast用于进行底层的位级转换,它可以将一种类型的指针转换为另一种类型的指针,但是需要谨慎使用,因为它可能会导致未定义行为。static_cast是一种通用的类型转换操作符,它可以进行标准的类型转换,例如将整数类型转换为浮点数类型,或者将指针类型转换为另一种指针类型。总之,C语言中的这四种强制类型转换操作符提供了灵活的类型转换能力,但需要根据具体的需求来选择合适的操作符。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [C++四种强制类型转换](https://blog.csdn.net/challenglistic/article/details/128882220)[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^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [深入C++四种强制类型转换的总结](https://download.csdn.net/download/weixin_38723192/14866840)[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^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值