C/C++中类型转换:static_cast, dynamic_cast, const_cast, reinterpret_cast

2024-01-05 02:52

本文主要是介绍C/C++中类型转换:static_cast, dynamic_cast, const_cast, reinterpret_cast,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 目的
    • 定义和作用

目的

在C中一般是使用(int) x, (int *)x, (void *) p等方式,但是这种强制类型转换的方式是不安全的,因为他没有类型检查。而且这种转换也是不够精确的。
因此在C++中引入了几种强制类型转换的方式。

定义和作用

  1. static_cast:

    1. 提供安全的基本类型转换:但是要注意数据的截断或者改变。(float转换为整数的时候丢失小数部分)。
    2. 类层次间转换:
      向上转换(将派生类的指针/引用转换为基类的指针/引用),这是安全的,因为派生类总是包含基类的全部信息。
      向下转换(将基类的指针/引用转换为派生类的指针/引用),这是不安全的,因为基类不一定包含派生类的全部信息。
    3. 空指针转换为目标类型的空指针:
      class Base {};
      class Derived : public Base {};Base* b = nullptr;
      Derived* d = static_cast<Derived*>(b);  // 安全转换,d 仍然是 nullptr。
      
    4. non_const转换为const
      int x = 10;
      const int* px = static_cast<const int*>(&x);
      
  2. dynamic_cast:
    用于多态,尤其是向下转换和侧向转换(在同一层次结构中的不同类之间)。
    在执行转换时进行运行时检查。如果是安全的(即对象确实是目标类型或其派生类型的实例),则转换成功;如果不安全则转换失败(对于指针,返回nullptr,对于引用抛出异常。

    class Base { virtual void dummy() {} };
    class Derived : public Base { /* ... */ };Base* basePtr = new Derived();// 安全的向下转换
    Derived* derivedPtr = dynamic_cast<Derived*>(basePtr);
    

    注意:

    1. 使用dynamin_cast必要要求设计的类至少有一个虚函数,因为涉及多态。
    2. 编译时无法确定转换结果,所以是运行时检查。

    补充侧向转换:
    指的是在同一继承层次中,将一个基类的指针或引用转换为另一个并行基类的指针或引用。
    这种转换通常发生在多重继承的情况下,当一个派生类继承自多个基类时。
    侧向转换的场景
    假设有一个多重继承的类层次结构,其中 Derived 类继承自两个基类 Base1 和 Base2:

    Copy code
    class Base1 { /* ... */ };
    class Base2 { /* ... */ };
    class Derived : public Base1, public Base2 { /* ... */ };
    

    在这种结构中,Derived 同时是 Base1 和 Base2 的子类。如果你有一个指向 Base1 的指针,且它实际上指向的是一个 Derived 类型的对象,你可以使用侧向转换将这个指针安全地转换为指向 Base2 的指针:

    Copy code
    Derived d;
    Base1* b1 = &d;
    Base2* b2 = dynamic_cast<Base2*>(b1); // 侧向转换
    

在这个例子中,dynamic_cast 被用来从 Base1 类型转换到 Base2 类型。由于这两个类是在同一个继承层次中,这样的转换是可能的,并且如果 b1 确实指向一个 Derived 类型的对象,转换将会成功。

  1. const_cast:
    去除对象指针/引用的const属性
    const_cast 是 C++ 中的一种类型转换操作符,专门用于修改对象的 constvolatile 属性。它是用来改变对象的常量性(const-ness)或易失性(volatility)的。
    使用场景与目的

    1. 移除 constvolatile 限定:

      • 当你有一个指向常量的指针或引用,但需要调用一个非常量成员函数或需要修改其指向的数据时,可以使用 const_cast 来移除 constvolatile 限定。
      • 这常用于与旧的C风格代码交互,其中数据不是常量,但接口要求传递常量参数。
    2. 添加 const 限定:

      • 虽然不常见,但 const_cast 也可以用来给对象添加 const 限定。这在某些特定情况下可能有用。
        使用示例
    const int a = 10;
    int* b = const_cast<int*>(&a);  // 移除 const 限定
    *b = 20;  // 未定义行为,因为 a 是一个真正的常量
    

    在此例中,尽管 const_cast 成功移除了 const 限定,但对 a 的修改是未定义行为,因为 a 本身是一个常量。

    局限(缺点)

    1. 未定义行为风险:
      如果原始对象实际上是一个常量,尝试修改它的值(即使使用 const_cast 移除了 const 限定)将导致未定义行为。

    2. 安全性问题:
      滥用 const_cast 可能导致代码难以理解和维护,因为它破坏了 const 正确性,这是 C++ 用于保证对象不被意外修改的机制之一。

    3. 限制用途:
      const_cast 只能改变对象的 const/volatile 限定,不能用于改变对象的实际类型。

    4. 破坏封装性:
      在类的上下文中,const_cast 可能会破坏类的封装性,因为它允许修改本应被保护的成员。

  2. reinterpret_cast:
    它用于在不兼容类型之间进行转换,基本上是直接重新解释底层的位模式。由于这种转换的低级性质,它通常在特定的系统编程和硬件相关的场景中使用。

    使用场景与目的

    1. 不兼容类型之间的转换:
      用于在完全不相关的类型之间进行转换,例如将指针转换为足够大的整数类型,或将不同类型的指针之间进行转换。
    2. 系统级或硬件相关编程:
      在底层编程中,如操作系统内核或硬件接口编程时,可能需要将指针或其他类型直接转换为某种特定格式。
    3. 函数指针的转换:
      用于将一个类型的函数指针转换为另一个类型。这在需要将函数指针传递给期望不同签名的函数指针参数的回调函数时特别有用。
      示例
    Copy code
    int* ip = new int(42);
    char* cp = reinterpret_cast<char*>(ip);  // 将 int* 转换为 char*
    

    在此示例中,ip 指向一个整数,使用 reinterpret_cast 将其转换为指向 char 的指针。

    局限(缺点)

    1. 类型安全问题:
      reinterpret_cast 不进行任何类型安全检查。它简单地重新解释给定值的位模式,这可能导致未定义行为,特别是在不恰当地解释数据时。
    2. 可移植性问题:
      由于它依赖于特定平台的底层数据表示方式,所以使用 reinterpret_cast 的代码通常缺乏可移植性。
    3. 维护性和可读性问题:
      这种转换的意图不够明确,可能导致代码难以理解和维护。
    4. 潜在的运行时错误:
      错误的使用 reinterpret_cast 可能导致程序崩溃、数据损坏或其他难以调试的运行时错误。
  3. 违反别名规则:
    在某些情况下,reinterpret_cast 可能违反 C++ 的严格别名规则(Strict Aliasing Rule),这可能导致编译器优化产生问题。

这篇关于C/C++中类型转换:static_cast, dynamic_cast, const_cast, reinterpret_cast的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/571435

相关文章

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑