运行时动态类型的识别其实应该是多态方面的知识,这里我直接拿来单独成章。

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++ 运行时类别识别的更多相关文章

  1. RTTI 运行时类型识别 及异常处理

    RTTI   运行时类型识别 typeid  ------  dynamic_cast dynamic_cast 注意事项: 1.只能应用于指针和引用之间的转化 2.要转换的类型中必须包含虚函数 3. ...

  2. RTTI (Run-Time Type Identification,通过运行时类型识别) 转

    参考一: RTTI(Run-Time Type Identification,通过运行时类型识别)程序能够使用基类的指针或引用来检查这些指针或引用所指的对象的实际派生类型.   RTTI提供了以下两个 ...

  3. MFC六大核心机制之二:运行时类型识别(RTTI)

    上一节讲的是MFC六大核心机制之一:MFC程序的初始化,本节继续讲解MFC六大核心机制之二:运行时类型识别(RTTI). typeid运算子 运行时类型识别(RTTI)即是程序执行过程中知道某个对象属 ...

  4. c++运行时类型识别(rtti)

    一个简单运行时类型识别 namespace rtti_ex { /* * 类型信息基类 */ class i_type_info { public: // 判断是否是指定类型 bool is(cons ...

  5. Java基础之RTTI 运行时类型识别

    运行时类型识别(RTTI, Run-Time Type Identification)是Java中非常有用的机制,在Java运行时,RTTI维护类的相关信息. 多态(polymorphism)是基于R ...

  6. 框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解)

    框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解) 一丶什么是RTTI,以及RTTI怎么设计 通过第一讲,我们知道了怎么样升成一个窗口了,以及简单的消息循环. 第二讲则是主要讲解RTTI ...

  7. MFC原理第三讲.RTTI运行时类型识别

    MFC原理第三讲.RTTI运行时类型识别 一丶什么是RTTI RTTI. 运行时的时候类型的识别. 运行时类型信息程序.能够使用基类(父类)指针 或者引用 来检查这些指针或者引用所指的对象. 实际派生 ...

  8. (C/C++学习笔记) 二十三. 运行时类型识别

    二十三. 运行时类型识别 ● 定义 运行时类型识别(Run-time Type Identification, RTTI) 通过RTTI, 程序能够使用基类的指针或引用来检查(check)这些指针或引 ...

  9. RTTI(运行时类型识别)

    运行时类型识别(Run-time type identification , RTTI),是指在只有一个指向基类的指针或引用时,确定所指对象的准确类型的操作.其常被说成是C++的四大扩展之一(其他三个 ...

随机推荐

  1. docker工具之基本命令

    docker工具之基本命令 1.docker服务的启动.停止.重启 systemctl start docker #启动docker服务 systemctl daemon-reload #守护进程重启 ...

  2. Python:requests库、BeautifulSoup4库的基本使用(实现简单的网络爬虫)

    Python:requests库.BeautifulSoup4库的基本使用(实现简单的网络爬虫) 一.requests库的基本使用 requests是python语言编写的简单易用的HTTP库,使用起 ...

  3. Luogu P2210 Haywire 题解

    其实这题吧...有一种玄学解法 这题的要求的就是一个最小化的顺序 那么,我们就不进想到了一种显然的写法 就是random_shuffle 什么?这不是乱搞的非正解吗 然而,正如一句话说的好 一个算法, ...

  4. 自学Java,需要掌握什么内容才能找到满意的工作?

    首先,这个问题主要问:自学Java编程技术,如果才能找到一份Java编程的工作.按照现在的招聘标准来看,无论你去哪个公司面试,你只需要满足他们公司的需求就可以. 找到一份Java编程工作需要掌握的内容 ...

  5. js 生成32位随机数,可用于微信支付流水号(前端生成)

    $(function () { /*生成32位随机流水号*/ /*默认去掉了容易混淆的字符oOLl,9gq,Vv,Uu,I1*/ var $chars = 'ABCDEFGHJKMNPQRSTWXYZ ...

  6. .netcore2.1 JS-SDK 从后台获取微信签名,实现自定义分享标题、描述、图片

    最近项目移动端需要实现微信自定义分享功能,包含分享自定义标题.描述等. 首先到公众号的后台,功能设置里面,添加将要被分享的域名,如图 后端签名算法实现 ,参考腾讯开发者文档https://mp.wei ...

  7. curl的使用指南

    一.查看网页源码 直接在curl命令后加上网址,就可以看到网页源码.我们以网址www.sina.com为例(选择该网址,主要因为它的网页代码较短): $ curl www.sina.com ​   M ...

  8. selenium时间等待方法

    在UI自动化测试中,必然会遇到环境不稳定.网络慢等情况.当你觉得定位没有问题,但程序却直接报了元素不可见时,那你就需要思考是否因为程序运行太快或者页面加载太慢而造成了元素不可见,必须要再等待直至元素可 ...

  9. MySQL数据库:在命令提示符中使用mysql

    服务启动 在命令提示符中 启动mysql服务 net start mysql 停止mysql服务 net sotp mysql 通过命令行进入 mysql -u 用户名 -p 键入后会提示输入密码 如 ...

  10. Linux—系统管理

    查看Linux系统版本                                                                    CentOS系统 # centos7查看命 ...