(C/C++学习笔记) 二十三. 运行时类型识别
二十三. 运行时类型识别
● 定义
|
运行时类型识别(Run-time Type Identification, RTTI) |
|
通过RTTI, 程序能够使用基类的指针或引用来检查(check)这些指针或引用所指对象的实际派生类型. C++通过下面两个操作符提供RTTI: ① typeid操作符, 返回指针或引用所指对象的实际类型; ② dynamic_cast操作符, 将基类类型的指针或引用安全地转换为派生类型的指针或引用. |
● 基类指针访问子类的特有成员函数
|
#include <iostream> using namespace std; class Base { public: virtual char* GetName()=0; }; class Bint:public Base { public: char * GetName() {return "Bint";} int GetInt() {return 1;} }; class BString:public Base { public: char* GetName() {return "BString";} char* GetString() {return "Hello";} }; int main(int argc, char* argv[]) { Base* B1 = new Bint(); //也可以写成Base* B1 = (Base*)new Bint(); 但这里的(Base*)其实是多余的 printf(B1->GetName()); printf("\n"); //////////////////////////////////////////////////////////////// Bint *B2= static_cast<Bint*>(B1); //下面要用基类指针B1来访问子类Bint的特有方法GetInt(), 我们首先要将B1静态转换为Bint型 if(B2) printf("%d", B2->GetInt()); printf("\n"); //////////////////////////////////////////////////////////////// Base * C1 = new BString(); printf(C1->GetName()); printf("\n"); //////////////////////////////////////////////////////////////// BString *C2 = static_cast<BString *>(C1); if(C2) printf(C2->GetString()); printf("\n"); return 0; }
|
● RTTI与虚继承
|
如果使用了虚继承, 将子类对象的地址赋值给基类指针, 此时, 基类指针仍可识别为它所指的子类对象的实际类型. ※ RTTI (Runtime Type Identification) 运行时类型识别 typeid 是一个用于运行时类型识别的操作符, 它返回的是type_info类型的常引用, type_info是C++标准库的一个类. 使用typeid注意使用头文件typeinfo.h |
|
#include <iostream> #include <typeinfo> using namespace std; class Animal { public: virtual void display(){} }; class Cow:public Animal{}; class Dog:public Animal {}; int main(){ Animal *anm; Cow cw; Dog dg; cout << "type of anm is : "<<typeid(anm).name()<<endl; cout << "type of cw is : "<<typeid(cw).name()<<endl; cout << "type of dg is : "<<typeid(dg ).name()<<endl; // initializing base pointer with the address of a derived class object; anm = &cw; cout<<"type of *anm when pointing to cw is: "<<typeid(*anm).name()<<endl; anm = &dg; cout<<"type of *anm when pointing to dg is: "<<typeid(*anm).name()<<endl; system("PAUSE"); return 0; } |
|
|
● 标准C++的类型转换符
|
对于强制类型转换的风险, 最好的解决方法就是不要使用C风格的强制类型转换,而是使用标准C++的类型转换符:static_cast、dynamic_cast、reinterpret_cast、和const_cast。 用法: dynamic_cast < typeid > ( expression ); 以此类推 ※ 所有的cast操作符都是针对 指针或引用 来转换的,不能对普通类型进行操作. ① static_cast 进行上行转换(把子类的指针或引用转换成基类表示)是安全的;进行下行转换(把基类指针或引用转换成子类表示)时,由于没有动态类型检查(check),所以是不安全的。 ② dynamic_cast 进行上行转换时,dynamic_cast和static_cast的效果是一样的;在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全 ③ reinterpret_cast 将某一类型转换为原来的类型 #include "iostream.h" int main(int argc, char* argv[]) { int n = 97; char p[4] = {0}; 个元素 cout<<p<<endl; 个元素)的地址转换成原来的整型指针 cout<<*f<<endl; return 0; }
④const_cast 强制将一个const变量(只读变量)变成一个非const的等价形式(即并不就是把某个const变量转换为一般的变量) #include <iostream> using namespace std; int main(int argc, char* argv[]) { const int a=1; int* b=const_cast<int*>(&a); //不能是int b=const_cast<int>(a) * b=2; //const变量a的等价形式是*b cout<<* b<<endl; }
|
● 数组和向量
|
在计算机领域, 一维数组(single-dimensional array)=向量(能自己管理内存,又不限长度); 二维数组(2-dimensional array)=矩阵 ※"之所以选择'vector'这个名字,是因为Alex Stepanov作为C++标准库的设计者当初在寻找一个可以与内置数组类型区分开来的名字的时候,使用了"向量"这个表示; 他现在承认这确实是个错误,因为数学家们通常使用"向量"来表示一个定长的序列。 |
● 向量
|
向量的定义方式有四种: vector<int> a(10); vector<int> b(10, 1); vector<int> c(b); vector<int> d(b.begin(), b.begin()+3);
如果要输出向量中的所有元素, 可以有两种循环控制方式: //第一种: for (int i=0; i<a.size(); i++) cout<<a[i]<<" "; //第二种: for (vector<int>::iterator it=a.begin(); it!=a.end(); ++it) cout<<*it<<" "; |
(C/C++学习笔记) 二十三. 运行时类型识别的更多相关文章
- MFC六大核心机制之二:运行时类型识别(RTTI)
上一节讲的是MFC六大核心机制之一:MFC程序的初始化,本节继续讲解MFC六大核心机制之二:运行时类型识别(RTTI). typeid运算子 运行时类型识别(RTTI)即是程序执行过程中知道某个对象属 ...
- python3.4学习笔记(二十三) Python调用淘宝IP库获取IP归属地返回省市运营商实例代码
python3.4学习笔记(二十三) Python调用淘宝IP库获取IP归属地返回省市运营商实例代码 淘宝IP地址库 http://ip.taobao.com/目前提供的服务包括:1. 根据用户提供的 ...
- C++学习之显式类型转换与运行时类型识别RTTI
static_cast const_cast reinterpret_cast 运行时类型识别(RTTI) dynamic_cast 哪种情况下dynamic_cast和static_cast使用的情 ...
- RTTI (Run-Time Type Identification,通过运行时类型识别) 转
参考一: RTTI(Run-Time Type Identification,通过运行时类型识别)程序能够使用基类的指针或引用来检查这些指针或引用所指的对象的实际派生类型. RTTI提供了以下两个 ...
- 框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解)
框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解) 一丶什么是RTTI,以及RTTI怎么设计 通过第一讲,我们知道了怎么样升成一个窗口了,以及简单的消息循环. 第二讲则是主要讲解RTTI ...
- MFC原理第三讲.RTTI运行时类型识别
MFC原理第三讲.RTTI运行时类型识别 一丶什么是RTTI RTTI. 运行时的时候类型的识别. 运行时类型信息程序.能够使用基类(父类)指针 或者引用 来检查这些指针或者引用所指的对象. 实际派生 ...
- 《深入浅出MFC》系列之运行时类型识别(RTTI)
/********************************************************************************** 发布日期:2017-11-13 ...
- RTTI 运行时类型识别 及异常处理
RTTI 运行时类型识别 typeid ------ dynamic_cast dynamic_cast 注意事项: 1.只能应用于指针和引用之间的转化 2.要转换的类型中必须包含虚函数 3. ...
- c++运行时类型识别(rtti)
一个简单运行时类型识别 namespace rtti_ex { /* * 类型信息基类 */ class i_type_info { public: // 判断是否是指定类型 bool is(cons ...
随机推荐
- boke练习: springboot整合springSecurity出现的问题,传递csrf
boke练习: springboot整合springSecurity出现的问题,传递csrf freemarker模板 在html页面中加入: <input name="_csrf&q ...
- python FileNotFoundError: [WinError 2] 系统找不到指定的文件。
D:\Test\TestSWP\Scripts\python.exe D:/Test/SWP/TestSWP/chorme/chorme.pyTraceback (most recent call l ...
- react-navigation学习笔记
1.关于this.props.navigation.navigate()与this.props.navigation.push()的区别 navigate方法在跳转时会在已有的路由堆栈中查找是否已经存 ...
- 前端VUE框架-es6
EMCAScript 6 又叫 es2015 1.常量和变量 常量: const a = "hello" 常量不能修改和重复定义 变量: let:定义一个块级作用域的变量 需要先定 ...
- Excel中如何匹配另外一个Excel中的数据
场景: 我在Excel中想展示通过一列匹配到另外Excel中的数据.对于程序员来说,就是left join 出 B表的数据. 但是在Excel中怎么做呢,我又不想每次都在把数据导入到数据库中操作. 这 ...
- 使用INTERSECT运算符
显示符合以下条件的雇员的雇员ID 和职务ID:这些雇员的当前职务与以前的职务相同,也就是说这些雇员曾担任过别的职务,但现在又重新担任了以前的同一职务. hr@TEST0924> SELECT e ...
- Beta阶段——第1篇 Scrum 冲刺博客
第1篇 Scrum 冲刺博客 a. 介绍小组新加入的成员,Ta担任的角色. 新加入成员 郭炜埕 原先担任的角色 前端界面设计 现在担任的角色 前端开发,并协助后端开发 新加成员介绍 炜埕同学对界面设计 ...
- Spring注解之 @SuppressWarnings注解
简介:java.lang.SuppressWarnings是J2SE5.0中标准的Annotation之一.可以标注在类.字段.方法.参数.构造方法,以及局部变量上.作用:告诉编译器忽略指定的警告,不 ...
- docker 系列之 docker安装
Docker支持以下的CentOS版本 CentOS 7 (64-bit) CentOS 6.5 (64-bit) 或更高的版本 前提条件 目前,CentOS 仅发行版本中的内核支持 Docker. ...
- Buffer和Stream
Buffer JavaScript 语言自身只有字符串数据类型,没有二进制数据类型.但在处理像TCP流或文件流时,必须使用到二进制数据. 因此在 Node.js中,定义了一个 Buffer 类,该类用 ...





