您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > JAVA

C++四种强制类型转换介绍

时间:2020-09-27 09:40:27  来源:  作者:
C++四种强制类型转换介绍

 

接上文:C语言的隐式类型转换和显示类型转换

隐式类型转换是编译器自动隐式进行的,需要在代码中体现,而显示类型转换由程序员明确指定。

C++支持C风格的强制转换,但是C风格的强制转换可能带来一些隐患,让一些问题难以发现。

所以C++提供了一组适用于不同场景的强制转换的函数:

  • static_cast
  • dynamic_cast
  • const_cast
  • reinterpret_cast

下面对这四种转换操作的适用场景分别进行说明。

1、static_cast

static_cast<type>(expression)

该运算符把 expression 转换为 type 类型,主要用于基本数据类型之间的转换,如把 uint 转换为 int,把 int 转换为 double 等。

uint x = 1;
int y = static_cast<int>(x); // 转换正确
int x = 1;
double y = static_cast<double>(x); // 转换正确

需要注意的是:static_cast 没有运行时类型检查来保证转换的安全性,需要程序员来判断转换是否安全。

int x = -1;
uint y = static_cast<uint>(x) // 转换错误
double x = 1.23;
int y = static_cast<int>(x) // 转换丢失精度

static_cast 还可用于类层次结构中,基类和派生类之间指针或引用的转换,但也要注意:

  • static_cast 进行上行转换是安全的,即把派生类的指针转换为基类的;
  • static_cast 进行下行转换是不安全的,即把基类的指针转换为派生类的。
// 上行转换,派生类→基类
Derive* d = new Derive();
Base* b = static_cast<Base*>(d);// 下行转换,基类→派生类
Base* b = new Base();
Derive* d = static_cast<Derive*>(b);

这是因为派生类包含基类信息,所以上行转换(只能调用基类的方法和成员变量),一般是安全的;

而基类没有派生类的任何信息,而下行转换后会用到派生类的方法和成员变量,这些基类都没有,很容易“指鹿为马”,或指向不存在的空间。

2、dynamic_cast

dynamic_cast<type>(expression)

dynamic_cast 主要用于类层次间的上行转换或下行转换。在进行上行转换时,dynamic_cast 和 static_cast 的效果是一样的,但在下行转换时,dynamic_cast 具有类型检查的功能,比 static_cast 更安全。

比如下面这段代码:

#include <IOStream>
using namespace std;
class Base {
public:
    virtual void Say() {
        cout << "I am Base." << endl;
    }};class Derive : public Base {
public:
    virtual void Say() {
        cout << "I am Derive." << endl;
    }};int main(){    // 上行转换
    Derive* d1 = new Derive();
    cout << "d1: " << d1 << endl;
    Base* b1 = dynamic_cast<Base*>(d1);
    cout << "b1: " << b1 << endl;
  
    // 下行转换
    Base* b2 = new Base();
    cout << "b2: " << b2 << endl;
    Derive* d2 = dynamic_cast<Derive*>(b2);
    cout << "d2: " << d2 << endl;
    return 0;
}

运行结果为:

C++四种强制类型转换介绍

 

在进行下行转换时,从基类 b2 到 d2 时,d2 会改为空指针(0x0),这正是 dynamic_cast 提升安全的功能。这个检查主要来自虚函数表。

在C++面向对象的思想中,虚函数是实现多态的关键机制。当一个类中有虚函数时,那么编译器就会构建出一个虚函数表来指示这些函数的地址。当用基类的指针指向派生类的对象,调用方法时就会根据虚函数表找到对应派生类的方法。

注意:B 要有虚函数,否则会编译出错;static_cast则没有这个限制。

这是由于运行时类型检查需要运行时类型信息,而这个信息存储在类的虚函数表,只有定义了虚函数的类才有虚函数表,没有定义虚函数的类是没有虚函数表的。

 

3、const_cast

const_cast<type>(expression)

该运算符用来修改 expression 的 const 或 volatile 属性。这里需要注意:expression 和 type 的类型一样的。

比如下面的代码,指针 px 由于有 const 修饰,无法直接通过其修改 x 的值,但又期望能修改 x 的值时,怎么办呢?这时就需要用到 const_cast。

int main()
{    int x = 1;
    cout << "before: " << x << endl;
      const int* px = &x;
    // *px = 2; // 编译错误
      int* py = const_cast<int*>(px);
    *py = 2;
      cout << "px: " << px << endl;
    cout << "py: " << py << endl;
    cout << "after : " << x << endl;
      return 0;
}

运行结果为:

C++四种强制类型转换介绍

 

可以看出,px 和 py 指向同一个地址,但通过 py 就可以修改 x 的值了。

这是因为通过const_cast,就把 const 类型的指针 px 转换成非 const 类型的指针 py 了。

4、reinterpret_cast

reinterpret_cast<type>(expression)

该运算符可以把一个指针转换成一个整数,也可以把一个整数转换成一个指针。

int main()
{    int* p = new int(5);
    uint64_t p_val = reinterpret_cast<uint64_t>(p);
      cout << "p    :" << p << endl;
    cout << "p_val:" << hex << p_val << endl;
      return 0;
}

上述代码把指针 p 的地址值转换成了 uint64_t 类型的整数值,运行结果为:

C++四种强制类型转换介绍

 

这个转换是“最不安全”的。不推荐使用。

综上,在使用强制类型转换时,需要首先考虑清楚使用目的,总结如下:

  • static_cast:基本类型转换,低风险;
  • dynamic_cast:类层次间的上行转换或下行转换,低风险;
  • const_cast:去 const 属性,低风险;
  • reinterpret_cast:转换不相关的类型,高风险。


Tags:C++ 类型转换   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
接上文:C语言的隐式类型转换和显示类型转换隐式类型转换是编译器自动隐式进行的,需要在代码中体现,而显示类型转换由程序员明确指定。C++支持C风格的强制转换,但是C风格的强制...【详细内容】
2020-09-27  Tags: C++ 类型转换  点击:(51)  评论:(0)  加入收藏
不同的数据在计算机内存中的存储方式不同,导致了“类型”这一抽象概念的出现。...【详细内容】
2019-09-24  Tags: C++ 类型转换  点击:(114)  评论:(0)  加入收藏
▌简易百科推荐
一、Redis使用过程中一些小的注意点1、不要把Redis当成数据库来使用二、Arrays.asList常见失误需求:把数组转成list集合去处理。方法:Arrays.asList 或者 Java8的stream流式处...【详细内容】
2021-12-27  CF07    Tags:Java   点击:(3)  评论:(0)  加入收藏
文章目录 如何理解面向对象编程? JDK 和 JRE 有什么区别? 如何理解Java中封装,继承、多态特性? 如何理解Java中的字节码对象? 你是如何理解Java中的泛型的? 说说泛型应用...【详细内容】
2021-12-24  Java架构师之路    Tags:JAVA   点击:(5)  评论:(0)  加入收藏
大家好!我是老码农,一个喜欢技术、爱分享的同学,从今天开始和大家持续分享JVM调优方面的经验。JVM调优是个大话题,涉及的知识点很庞大 Java内存模型 垃圾回收机制 各种工具使用 ...【详细内容】
2021-12-23  小码匠和老码农    Tags:JVM调优   点击:(11)  评论:(0)  加入收藏
前言JDBC访问Postgresql的jsonb类型字段当然可以使用Postgresql jdbc驱动中提供的PGobject,但是这样在需要兼容多种数据库的系统开发中显得不那么通用,需要特殊处理。本文介绍...【详细内容】
2021-12-23  dingle    Tags:JDBC   点击:(12)  评论:(0)  加入收藏
Java与Lua相互调用案例比较少,因此项目使用需要做详细的性能测试,本内容只做粗略测试。目前已完成初版Lua-Java调用框架开发,后期有时间准备把框架进行抽象,并开源出来,感兴趣的...【详细内容】
2021-12-23  JAVA小白    Tags:Java   点击:(10)  评论:(0)  加入收藏
Java从版本5开始,在 java.util.concurrent.locks包内给我们提供了除了synchronized关键字以外的几个新的锁功能的实现,ReentrantLock就是其中的一个。但是这并不意味着我们可...【详细内容】
2021-12-17  小西学JAVA    Tags:JAVA并发   点击:(10)  评论:(0)  加入收藏
一、概述final是Java关键字中最常见之一,表示“最终的,不可更改”之意,在Java中也正是这个意思。有final修饰的内容,就会变得与众不同,它们会变成终极存在,其内容成为固定的存在。...【详细内容】
2021-12-15  唯一浩哥    Tags:Java基础   点击:(14)  评论:(0)  加入收藏
1、问题描述关于java中的日志管理logback,去年写过关于logback介绍的文章,这次项目中又优化了下,记录下,希望能帮到需要的朋友。2、解决方案这次其实是碰到了一个问题,一般的情况...【详细内容】
2021-12-15  软件老王    Tags:logback   点击:(17)  评论:(0)  加入收藏
本篇文章我们以AtomicInteger为例子,主要讲解下CAS(Compare And Swap)功能是如何在AtomicInteger中使用的,以及提供CAS功能的Unsafe对象。我们先从一个例子开始吧。假设现在我们...【详细内容】
2021-12-14  小西学JAVA    Tags:JAVA   点击:(21)  评论:(0)  加入收藏
一、概述观察者模式,又可以称之为发布-订阅模式,观察者,顾名思义,就是一个监听者,类似监听器的存在,一旦被观察/监听的目标发生的情况,就会被监听者发现,这么想来目标发生情况到观察...【详细内容】
2021-12-13  唯一浩哥    Tags:Java   点击:(16)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条