0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

RTTI和类型转换运算符

冬至子 来源:iDoitnow 作者:艰默 2023-07-18 11:23 次阅读

1. RTTI

RTTI是 运行阶段类型识别 (Runtime Type Identification)的简称。该特性是为了程序在运行阶段确定对象类型提供一种标准方式。

1.1 RTTI的三个元素

由于只有包含虚函数的类层次结构,才能将派生类对象的地址赋给基类指针,因此RTTI只适用包含虚函数的类。

  • dynamic_cast运算符将使用一个指向基类的指针来生成一个指向派生类的指针,否则,该运算符返回0—空指针。
  • typeid运算符返回一个指出对象的类型的值。
  • type_info结构存储了有关特定类型的信息

1.1.1 dynamic_cast运算符

dynamic_cast运算符是RTTI中最常用的组件,其主要的功能是 确定是否可以安全地将对地址赋给特定类型的指针 ,如果可以则返回该特定类型的指针,如果不可以则返回0(空指针)。可以理解为dynamic_cast可以用来类型转换,如果可安全转换,则转换成相应的类型指针,否则直接返回0(空指针)。其语法如下:

dynamic_cast< type_name > (expression)

例如下面的例子:

class Base{
    private:
    double val;
    public:
    Base(double a = 0):val(a){}
    virtial void show(){std::cout< < "Base show!"< < std::endl();}
}

class BaseA:public: Base{
    BaseA(double a = 0):Base(a){}
    virtual void show(){std::cout < < "BaseA show!"<

假设有下面的指针:

Base* bs = new Base;
Base* ba = new BaseA;
Base* bb = new BaseB;

对于下面类型的转换:

BaseB* b1 = dynamic_cast

因此针对该例子,dynamic_cast常用的场景如下:

auto bs = new BaseX;//BaseX为Base、BaseA、BaseB中的任意一种类型,假设其具体类型只有在程序运行过程中才能确定。
BaseA* pa;
if(pa = dynamic_cast

1.1.2 typeid运算符

typeid运算符能够确定两个对象是否为同种类型,其可以接受两种参数

  • 类名;
  • 结果为对象的表达式。

其返回一个type_info对象的引用,其实方法如下:

//需要包含头文件#include< typeinfo >
A* a = new A;//A为一个类
if(typeid(A) == typeid(*a)){
    std::cout < < "a的类型为A"<

针对上例,若a是一个空指针,则typeid(*a)会引发bad_typeid异常,该异常类型是从exception类派生出来,也是在typeinfo中生声明的。

1.1.3. type_info类

type_info类主要存储了有关特定类型的信息,其中包含了一个name()成员,该成员函数主要用于调试,其需要与typeid搭配使用,使用方法如下:

//class A;
A* a = new A;
std::cout< < typeid(*a).name()<

2. 类型转换运算符

2.1 dynamic_cast

前面已经介绍过其用法,该运算符的主要用途是,使得 能够在类层次结构中进行向上转换 (即派生类到基类的转换,由于该转换是is-a的关系,所以该转换是安全的),而不允许其他的转换。

2.2 const_cast

const_cast运算符用于执行只有一种用途的类型转化,即改变constvolatile,其语法与dynamic_cast相同:

const_cast < type_name > (expression)

如果类型的其他方面也被修改,则上述类型转换将出错,也就是说除了constvolatile特征(有或无)可以不同外,type_nameexpression的类型必须相同。假设AB,则有下面的转换:

A a;
const A* pa = &a;
A* pa1 = const_cast< A* >(pa); //正确
const B* pb = const_cast< const B* >(pa);//错误

提供该运算符的原因是,程序有时候可能需要一个这样的值,它在大多数的时候是常量,而有时候又是可以修改的,在这种情况下,可以将这个值声明为const,并在需要修改它的时候,使用const_cast

2.3 static_cast

static_cast运算符与其他类型转换运算符的语法一样:

static_cast < type_name > (expression)

static_castexpression转换为type_name类型,主要用于非多态类型之间的转换,不提供运行时的检查来确保转换的安全性。主要在以下几种场合中使用:

  • 用于类层次结构中 ,基类和子类之间指针和引用的转换;当进行上行转换(把派生类的指针或引用转换成基类表示),这种转换是安全的;当进行下行转换(把基类的指针或引用转换成派生类表示),这种转换是不安全的,也需要程序员来保证;
  • 用于基本数据类型之间的转换 ,如把int转换成char,把int转换成enum等等,这种转换的安全性需要程序员来保证;
  • 把void指针转换成目标类型的指针 ,是极其不安全的,也需要程序员来保证;

2.4 reinterpret_cast

reinterpret_cast运算符与其他类型转换运算符的语法一样:

reinterpret_cast < type_name > (expression)

reinterpret_cast运算符用于天生危险的类型转换,因为其允许将任何指针类型转换为其它的指针类型。reinterpret_cast运算符并不会改变括号中运算对象的值,而是对该对象从位模式上进行重新解释 。它主要用于将一种数据类型从一种类型转换为另一种类型。

例如,它可以将一个指针转换成一个整数,也可以将一个整数转换成一个指针,然而,其并不支持所有类型的转换,例如,可以将指针类型转换为足以存储指针表示的整形,但不能将指针转换为更小的整形或浮点型。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 转换器
    +关注

    关注

    27

    文章

    8217

    浏览量

    142163
  • 存储器
    +关注

    关注

    38

    文章

    7158

    浏览量

    162122
  • 运算符
    +关注

    关注

    0

    文章

    163

    浏览量

    10951
收藏 人收藏

    评论

    相关推荐

    JAVA语言的运算符及表达式

    JAVA语言的运算符及表达式3.5 运算符及表达式数据的类型除了限定数据的存储方式、取值范围之外,还定义了对该数据类型可进行的操作即运算。表
    发表于 12-06 00:28

    数据类型运算符与表达式

    数据类型运算符与表达式2.1   程序设计概述2.2  C语言的数据类型 2.3  常量和变量 2.4 
    发表于 03-10 15:05

    运算符

    浮点型变量可以存储实数字符型变量可以存储单个字符,其值是该字符的ASCII 码算术运算符提供运算功能,包括+、-、*、/、%、++、--printf() 和 scanf() 函数属于格式输入输出函数getchar() 和 putchar() 函数用来输入输出单个字符的函数
    发表于 06-20 16:40

    Verilog 缩减运算符

    小弟刚学FPGA ,在看Veriolg ,遇到缩减运算符,对于缩减与 缩减或能理解,那个非本身就是单目运算,缩减非是怎么回事呢?
    发表于 10-07 10:13

    C语言单片机第七课 运算符和表达式(关系运算符)

    第七课 运算符和表达式(关系运算符)关系运算符,同样我们也并不陌生。单片机C语言中有六种关系运算符,这些东西同样是在我们小时候学算术时就已经学习过了的:> 大于< 小于>= 大于等于<
    发表于 10-21 09:38

    STM32学习及应用笔记二:一次运算符优先级造成的错误

    ->成员名 ++后置自增运算符++变量名单目运算符 --后置自减运算符--变量名单目运算符 2-负号运算符-表达式右到左单目
    发表于 06-03 12:18

    【FPGA】VHDL 语言的运算符有哪些?计算的优先级是怎样的?

    的逻辑运算符如表 2-6 所示。2.关系运算符关系运算符两边必须为相同的类型,其结果为 boolean 类型。等号(=)和不等号(/=)两
    发表于 09-12 09:51

    【梦翼师兄今日分享】 常见的关系运算符(移位运算符)

    立即学习>>梦翼师兄FPGA培训(视频加板卡),手把手带你入门FPGA写在前面的话移位运算符是双目运算符,将运算符左边的操作数左移或右移指定的位数,用0来补充空闲位。如果右边操作数的值
    发表于 12-17 10:45

    【梦翼师兄今日分享】 常见的关系运算符(位拼运算符

    立即学习>>梦翼师兄FPGA培训(纯视频),手把手带你入门FPGA写在前面的话位拼运算符是将多个小的表达式合并形成一个大的表达式,用符号{}来实现多个表达式的连接运算,各个表达式之间用
    发表于 12-19 09:38

    C语言运算符优先级(超详细)

    !逻辑非运算符!表达式(类型)强制类型转换(数据类型)表达式--sizeof长度运算符sizeo
    发表于 04-27 16:47

    Java基础之Java运算符

    (n+"是奇数");} }}逻辑运算符用来连接一个或多个条件,并判断这些条件是否成立。逻辑运算符的结果是布尔类型。注意:逻辑运算符的操作数都是布尔
    发表于 05-18 15:39

    如何去使用三目运算符

    三目运算的定义是什么?三目运算符是由什么组成的?如何去使用三目运算符
    发表于 07-15 13:13

    算术运算符的相关资料分享

    一:算术运算符算术运算符非常地简单,就是小学数学里面的一些加减乘除操作。不过呢,还是有一些语法细节需要注意的。1.加法运算符 + 1 在第3行利用加法运算符 + 进行了加法
    发表于 11-30 06:09

    运算符的相关资料推荐

    运算符1、算数操作运算符+、-、*、/、%加法:A+B, AB最好是同类型乘除:乘法在很多CPU中并不支持,乘除法能不用就不用,可能会使执行变差求模/求余数:n%m=res[0~(m-1)]求模
    发表于 12-24 06:13

    2.7 python运算符

    2.7 python运算符0. 什么是运算符?本章节主要说明Python的运算符。举个简单的例子 4 +5 = 9 。 例子中,4和5被称为操作数,“+”号为运算符。Python语言支
    发表于 02-21 16:43