类型转换是一种机制,让程序员能够暂时或永久性改变编译器对对象的解释。注意,这并不意味着程序员改变了对象本身,而只是改变了对对象的解释

在很多情况下,类型转换是合理的需求,可解决重要的兼容问题。因此,程序员经常需要让编译器按其所需的方式解释数据,让应用程序能够成功编译并执行。

C++编译器仍需向后兼容以确保遗留代码能够通过编译,因此支持下面这样的语法:int* pBuf = (int *)pString ;

C风格类型转换实际上强迫编译器根据程序员的选择来解释目标对象,强迫编译器遵从自己的意愿。然而,对不希望类型转换破坏其倡导的类型安全的C++程序员来说,这是无法接受的。

C++提供了一种新的类型转换运算符,专门用于继承的情形,这种情形在C语言中并不存在。

4个C++类型运算符如下:

static_cast

dynamic_cast

reinterpret_cast

const_cast

1、static_cast

①用于相关类型的指针之间的转换

这个相关类型指针是指:具有继承关系的类的指针。

static_cast实现了基本的编译阶段检查,确保指针被转换为相关类型。使用static_cast可将指针向上转换为基类类型,也可向下转换为派生类类型。

这改进了C风格类型转换,在C语言中可将指向一个对象的指针转换为完全不相关的类型,而编译器不会报错。(C语言十分自由,而C++对于不同用途的类型转换,会有一些不同的约束)

注意:static_cast只验证指针类型是否相关(是否有继承关系类的指针),而不执行任何运行阶段检查。(故称之为静态static转换)

例:

CBase*pBase = new CBase( ) ;

CDerived*pDerived = static_cast<CDerived*>(pBase) ;

由于static_cast只在编译阶段检查转换类型是否相关,而不执行运行阶段检查,因此上面代码能够编译通过,但在运行阶段可能导致意外结果。

static_cast 与C风格的类型转换类似,只是static_cast只用于有继承相关性的类的指针。它会对这个关系进行检查。

故:

用static_cast把指向派生类的基类指针 转换为派生类指针,底层的派生类对象不变,指针的值也不变,只是改变了指针的管辖范围。即:只是改变了编译器的解释方式。(基类指针实际上指向的只是派生类中的基类核心)

同理,用static_cast把指向派生类的派生类指针 转换为基类指针,也只是改变了编译器对其的解释方式。(static_cast对与多重继承的指针,有特殊处理)

②用于内置数据类型的类型转换

例:

double dPi = 3.14;

int nNum =static_cast<int>(dPi) ;

使用static_cast可让代码阅读者注意到这里使用了类型转换,并指出编译器根据编译阶段可用信息进行了必要的调整,以便执行所需的类型转换。

2、dynamic_cast

dynamic_cast动态类型转换在运行阶段执行类型转换。可检查dynamic_cast操作的结果以判断类型转换是否成功。

dynamic_cast可以把基类指针转换为派生类指针,也可以把派生类指针转换为基类指针。如果转换后可以安全使用,则转换成功,否则其返回NULL。(而static_cast的转换,无法检查转换后的结果是否可以安全使用)

若基类指针指向的是基类对象,则把它dynamic_cast为派生类指针,则转换会失败。因为这是不安全的转换。

若基类指针指向的是派生类对象,则把它dynamic_cast为派生类指针,则转换会成功。这是安全的转换。(dynamic_cast比static_cast检查更加严格,但使用范围更小)

【注意】dynamic_cast必须要在有虚函数的继承里进行。(static_cast则无此限制)

给定一个基类指针,程序员可能不确定,它目前指向哪种类型,这时可使用dynamic_cast在运行时判断其类型,并在安全时使用转换后的指针。

dynamic_cast这种在运行阶段识别对象类型的机制成为 运行阶段类型识别RTTI。

关于RTTI

RTTI是Runtime Type Information的缩写,从字面上来理解就是执行期的类型信息,其重要作用就是动态判别执行期的类型。

即:判断基类指针或引用,目前所绑定的类型。

它有两种方法来识别:

①用dynamic_cast类型转换是否成功来识别类型。(dynamic_cast必须要在有虚函数的继承里进行)

例:

void fun(CBase* pBase)
{
CDerived* pDerived = dynamic_cast<CDerived*>(pBase) ;
if (pDerived != NULL)
pDerived->funDerived ;
else
pDerived->funBase ;
}

②用typeid判断基类地址是否一致来识别类型

例:

void fun(CBase* pBase)
{
CDerived* pDerived = NULL ;
if (typeid(*pBase) == typeid(CDerived))
{
pDerived = static_cast<CDerived*>(pBase) ;
pDerived->funDerived ;
}
else
pDerived->funBase ;
}

编译器从类中虚指针指向的虚函数表的前一表项中提取typeid的值。(请看 C++对象模型简介

【一般地讲,能用虚函数解决的问题就不要用“dynamic_cast”,能够用“dynamic_cast”解决的问题就不要用“typeid”。】

RTTI破坏了面向对象的纯洁性。

首先,它破坏了抽象。用RTTI检测当前基类指针绑定的类型,这与虚函数提倡的隐藏细节,智能实现多态相违背。

其次,因为运行时类型的不确定性,它把程序变得更脆弱。

第三,它使程序缺乏扩展性。当你在继承关系中加入了一个新类型时,你可能需要修改涉及RTTI的代码。

static_cast AND dynamic_cast的更多相关文章

  1. static_cast 和 dynamic_cast 的区别

    static_cast一般用来将枚举类型转换成整型,或者整型转换成浮点型.也可以用来将指向父类的指针转换成指向子类的指针.做这些转换前,你必须确定要转换的数据确实是目标类型的数据,因为static_c ...

  2. c++中的强制转换static_cast、dynamic_cast、reinterpret_cast的不同用法儿

    c++中的强制转换static_cast.dynamic_cast.reinterpret_cast的不同用法儿   虽然const_cast是用来去除变量的const限定,但是static_cast ...

  3. C++强制类型转换:static_cast、dynamic_cast、const_cast、reinterpret_cast

    1. c强制转换与c++强制转换 c语言强制类型转换主要用于基础的数据类型间的转换,语法为: (type-id)expression//转换格式1 type-id(expression)//转换格式2 ...

  4. C++中static_cast和dynamic_cast强制类型转换

    在C++标准中,提供了关于类型层次转换中的两个关键字static_cast和dynamic_cast. 一.static_cast关键字(编译时类型检查) 用法:static_cast < ty ...

  5. C++的类型转换:static_cast、dynamic_cast、reinterpret_cast和const_cast

    在C++中,存在类型转换,通常意味着存在缺陷(并非绝对).所以,对于类型转换,有如下几个原则:(1)尽量避免类型转换,包括隐式的类型转换(2)如果需要类型转换,尽量使用显式的类型转换,在编译期间转换( ...

  6. 四种强制类型转换的总结(const_cast、static_cast、dynamic_cast、reinterpreter_cast)

    四种强制类型转换的总结(const_cast.static_cast.dynamic_cast.reinterpreter_cast) 转载 2011年10月03日 23:59:05 标签: stru ...

  7. static_cast、dynamic_cast、reinterpret_cast、和const_cast

    关于强制类型转换的问题,很多书都讨论过,写的最详细的是C++ 之父的<C++ 的设计和演化>.最好的解决方法就是不要使用C风格的强制类型转换,而是使用标准C++的类型转换符:static_ ...

  8. static_cast 和 dynamic_cast

    1.static_cast static_cast < type-id > ( expression ) 该运算符把expression转换为type-id类型,但没有运行时类型检查来保证 ...

  9. C++ ------ static_cast,dynamic_cast,reinterpret_cast,const_cast

    C++类型转换分为:隐式类型转换和显式类型转换 第1部分. 隐式类型转换 又称为“标准转换”,包括以下几种情况:1) 算术转换(Arithmetic conversion) : 在混合类型的算术表达式 ...

  10. static_cast、dynamic_cast、const_cast和reinterpret_cast总结(转)

    前言 这篇文章总结的是C++中的类型转换,这些小的知识点,有的时候,自己不是很注意,但是在实际开发中确实经常使用的.俗话说的好,不懂自己写的代码的程序员,不是好的程序员:如果一个程序员对于自己写的代码 ...

随机推荐

  1. centos 6.5 配置nginx环境

    1.卸载系统中默认的php和httpd [root@x ~]# yum remove httpd* php* Loaded plugins: fastestmirror Setting up Remo ...

  2. java设计模式——桥接模式

    一. 定义与类型 定义:将抽象部分与他的具体实现部分分离,使它们都可以独立的变化,通过组合的方式建立两个类之间的联系,而不是继承 类型:结构性. 二. 使用场景 (1) 抽象和具体实现之间增加更多的灵 ...

  3. 微信端H5页面问题总结

    1.div元素不确定高度的情况下背景图片显示问题,解决后可以自适应不同分辨率的屏幕大小,div元素的background-size设置100%后,其自身的高度和宽度不能再设置. .register-t ...

  4. scrollHeight, scrollTop, clientHeight

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. Spring Cloud 升级最新 Finchley 版本,踩坑指南!

    https://blog.csdn.net/youanyyou/article/details/81530240 Spring Cloud 升级最新 Finchley 版本,踩了所有的坑! 2018年 ...

  6. 图解HTTP总结(4)——返回结果的HTTP状态码

    HTTP状态码负责表示客户端HTTP请求的返回结果.标记服务器端的处理是否正常.通知出现的错误等工作. 状态码的类别 2XX 成功 200 OK 表示从客户端发来的请求在服务器端被正常处理了. 在响应 ...

  7. Django项目发布到Apache2.4配置mod_wsgi,解决遭遇的各种坑。

    环境: Apache2.4 32bit Python 3.7.1 (v3.7.1:260ec2c36a, Oct 20 2018, 14:05:16) [MSC v.1915 32 bit (Inte ...

  8. 005--Django2.0的路由层

    URL配置就像Django所支撑的网站目录,它的本质是每条URL调用的视图函数的映射表,每一个请求执行对应的视图函数. 1.简单的路由配置  from django.contrib import ad ...

  9. POJ:2236-Wireless Network

    Wireless Network Time Limit: 10000MS Memory Limit: 65536K Total Submissions: 34265 Accepted: 14222 D ...

  10. 笔记-mysql-管理及基础操作

    笔记-mysql使用-管理及基础操作 1.      简介 mysql是一个免费的关系型数据库,不过好像被oracle收购了.... 希望它继续免费. 1.1.    相关术语 数据库,表,列,行,冗 ...