C++ 运行时类别识别
运行时动态类型的识别其实应该是多态方面的知识,这里我直接拿来单独成章。
dynamic_cast和static_cast
1.static_cast用法如下:
static_cast < Type-id > ( expression )
该运算符把expression转换为type-id类型,但没有运行时类型检查来保证转换的安全性。它主要可用于以下几种情况:
(1)用于类层次结构中基类和派生类之间指针或引用的转换。进行上行转换(把派生类的指针或引用转换成基类表示)是安全的;进行下行转换(把基类指针或引用转换成派生类表示)时,由于没有动态类型检查,所以是不安全的。
class Base { };
class Derived:public Base {};
int main()
{
Derived D;
Base* B = static_cast<Base*> (&D);
return ;
}
将派生类型的指针转化为基类型的指针
(2)用于基本数据类型之间的转换,如把int转换成char,把int转换成enum。这种转换的安全性也要开发人员来保证。
double a;
int b=;
a = static_cast<double> (b);
(3)把空指针转换成目标类型的空指针(不安全!!)。
(4)把任何类型的表达式转换成void类型。
C++提供了dynamic_cast操作符,可以在运行期间检测类型转换是否安全。dynamic_cast和static_cast有同样的语法:
dynamic_cast < Type-id > ( expression )
The dynamic_cast operator in C++ is used for downcasting a reference or pointer to a more specific type in the class hierarchy. Unlike the static_cast, the target of the dynamic_cast must be a pointer or reference to class.
Type-id必须是一个类的的指针或引用,也可以是 void *,参数expression必须是一个能得到一个指针或者引用的表达式。
同时,dynamic_cast仅对多态类型有效,也就是说使用dynamic_cast时要求基类中要有虚函数,否则会有编译错误;
而static_cast则没有这个限制。另外dynamic_cast要求转型的目的类型必须是指针或者引用。这是由于运行时类型检查需要运行时类型信息,而这个信息存储在类的虚函数表中,只有定义了虚函数的类才有虚函数表,没有定义虚函数的类是没有虚函数表的。
例:
class Base
{
public:
void fun( ) {};
};
class Derived:public Base { };
int main( )
{
Base b;
Derived d;
Base *pb = &b;
Derived *pd1 = static_cast<Derived *> (pb);
Derived *pd2 = dynamic_cast<Derived *>(pb); // error C2683: “dynamic_cast”:“Base”不是多态类型
return ;
}
若将fun()函数声明为虚函数,则不会有上面的错误。
如果pb实际指向一个 Derived类型的对象,pd1和pd2是一样的,并且这两个指针执行Derived类型的任何操作都是安全的;如果pb实际指向的是一个Base类型的对象,那么pd1将是一个指向该对象的指针,对它进行Derived类型的操作将是不安全的,而pd2将会得到一个空指针(即0,因为dynamic_cast失败)!
例:
#include <iostream>
using namespace std;
class Base
{
public:
virtual void fun1()
{
cout << "Base::fun1()" << endl;
}
virtual ~Base() { }
};
class Derived1: public Base
{
public:
virtual void fun1()
{
cout << "Derived1::fun1()" << endl;
}
virtual void fun2()
{
cout << "Derived1::fun2()" << endl;
}
};
class Derived2: public Derived1
{
public:
virtual void fun1()
{
cout << "Derived2::fun1()" << endl;
}
virtual void fun2()
{
cout << "Derived2::fun2()" << endl;
}
};
void fun(Base *b)
{
b->fun1();
//尝试将b转换为Derived1指针
Derived1 *d = dynamic_cast<Derived1 *>(b);
//判断转换是否成功
if (d != )
d->fun2();
}
int main()
{
Base b;
fun(&b);
Derived1 d1;
fun(&d1);
Derived2 d2;
fun(&d2);
return ;
}
运行结果:
Base::fun1()
Derived1::fun1()
Derived1::fun2()
Derived2::fun1()
Derived2::fun2()
typeid 获取运行时类型信息
typeid语法形式:
typeid ( 表达式 )
typeid ( 类型说明符 )
typeid功能
获得表达式或类型说明符的类型信息,表达式有多态类型时,会被求值,并得到动态类型信息;否则,表达式不被求值,只能得到静态的类型信息。
类型信息用type_info对象表示,type_info是typeinfo头文件中声明的类;
typeid的结果是type_info类型的常引用;
可以用type_info的重载的“==”、“!=”操作符比较两类型的异同;
type_info的name成员函数返回类型名称,类型为const char *。
#include <iostream>
#include <typeinfo>
using namespace std;
class Base
{
public:
virtual ~Base() { }
};
class Derived: public Base { };
void fun(Base *b)
{
const type_info &info1 = typeid(b);
const type_info &info2 = typeid(*b);
cout<<"typeid(b): "<<info1.name ()<<endl;
cout<<"typeid(*b): "<<info2.name ()<<endl;
if(info2 == typeid(Base))
cout<<"A base class!"<<endl;
else
cout<<"A Derived class!"<<endl;
cout<<endl;
}
int main()
{
Base b;
fun(&b);
Derived d;
fun(&d);
return ;
}
运行结果:
typeid(b): P4Base
typeid(*b): 4Base
A base class!
typeid(b): P4Base
typeid(*b): 7Derived
A Derived class!
C++ 运行时类别识别的更多相关文章
- RTTI 运行时类型识别 及异常处理
RTTI 运行时类型识别 typeid ------ dynamic_cast dynamic_cast 注意事项: 1.只能应用于指针和引用之间的转化 2.要转换的类型中必须包含虚函数 3. ...
- RTTI (Run-Time Type Identification,通过运行时类型识别) 转
参考一: RTTI(Run-Time Type Identification,通过运行时类型识别)程序能够使用基类的指针或引用来检查这些指针或引用所指的对象的实际派生类型. RTTI提供了以下两个 ...
- MFC六大核心机制之二:运行时类型识别(RTTI)
上一节讲的是MFC六大核心机制之一:MFC程序的初始化,本节继续讲解MFC六大核心机制之二:运行时类型识别(RTTI). typeid运算子 运行时类型识别(RTTI)即是程序执行过程中知道某个对象属 ...
- c++运行时类型识别(rtti)
一个简单运行时类型识别 namespace rtti_ex { /* * 类型信息基类 */ class i_type_info { public: // 判断是否是指定类型 bool is(cons ...
- Java基础之RTTI 运行时类型识别
运行时类型识别(RTTI, Run-Time Type Identification)是Java中非常有用的机制,在Java运行时,RTTI维护类的相关信息. 多态(polymorphism)是基于R ...
- 框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解)
框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解) 一丶什么是RTTI,以及RTTI怎么设计 通过第一讲,我们知道了怎么样升成一个窗口了,以及简单的消息循环. 第二讲则是主要讲解RTTI ...
- MFC原理第三讲.RTTI运行时类型识别
MFC原理第三讲.RTTI运行时类型识别 一丶什么是RTTI RTTI. 运行时的时候类型的识别. 运行时类型信息程序.能够使用基类(父类)指针 或者引用 来检查这些指针或者引用所指的对象. 实际派生 ...
- (C/C++学习笔记) 二十三. 运行时类型识别
二十三. 运行时类型识别 ● 定义 运行时类型识别(Run-time Type Identification, RTTI) 通过RTTI, 程序能够使用基类的指针或引用来检查(check)这些指针或引 ...
- RTTI(运行时类型识别)
运行时类型识别(Run-time type identification , RTTI),是指在只有一个指向基类的指针或引用时,确定所指对象的准确类型的操作.其常被说成是C++的四大扩展之一(其他三个 ...
随机推荐
- Spring afterPropertiesSet方法
1.init-method方法,初始化bean的时候执行,可以针对某个具体的bean进行配置.init-method需要在applicationContext.xml配置文档中bean的定义里头写明. ...
- js 运算的内置函数
// 一.Math.round()作用:四舍五入返回整数.(返回参数+0.5后,向下取整) // Math.round(5.57) //返回6 // Math.round(2.4) //返回2 // ...
- MySql索引背后的数据结构及算法
本文以MySQL数据库为研究对象,讨论与数据库索引相关的一些话题.特别需要说明的是,MySQL支持诸多存储引擎,而各种存储引擎对索引的支持也各不相同,因此MySQL数据库支持多种索引类型,如BTree ...
- Python3字典update()方法
描述 Python字典update()函数把字典参数dict2的key/value(键/值)对更新到字典dict里. update()方法语法: dict.update(dict2) 参数 dict2 ...
- ubuntu18.04 安装 WPS 2019
ubuntu自带的文字处理软件对来自windows下office或在WPS创建的ppt有点不兼容,看到WPS有linux版本的,便果断安装试一试. 一.卸载原生liboffice sudo apt-g ...
- Quartz.NET总结(三)如配置jobs
前两篇文章,已经介绍了Quartz.NET的使用和Cron表达式表达式的写法,今天说一说Quartz的配置,Quartz相关的配置有三个quartz.config.quartz_jobs.xml.lo ...
- .NET轻松实现支付宝服务窗网页授权并获取用户相关信息
前言: 最近在开发一个商业街区的聚合扫码支付功能,其中需要用到的有支付宝,微信两种支付方式,当然对于开发微信支付而已作为自己的老本行已经比较熟悉了,然而对于我来说支付宝支付还是头一次涉及到.这次项目中 ...
- PhantomJS抓取中文文档乱码
使用PhantomJs抓取百度标题乱码 var url="http://www.baidu.com" var page=require('webpage').create() pa ...
- Python中为什么不能用可变对象作为默认参数的值
def func(numbers = [], num=1): numbers.append(num) for number in numbers: print(number) func() >& ...
- 蓝色大气简约立体答辩ppt模板推荐
小编个人非常喜欢这个模版,大气深蓝色,具有科技感,非常适合学生的毕业答辩PPT模板. 模版来源:http://ppt.dede58.com/gongzuohuibao/26496.html