(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 ...
随机推荐
- ROS的安装和卸载
Robot Operating System (ROS) 是一个得到广泛应用机器人系统的软件框架,它包含了一系列的软件库和工具用于构建机器人应用.从驱动到最先进的算法,以及强大的开发者工具,ROS 包 ...
- 雷林鹏分享:jQuery EasyUI 窗口 - 自定义窗口工具栏
jQuery EasyUI 窗口 - 自定义窗口工具栏 默认情况下,窗口(window)有四个工具:collapsible.minimizable.maximizable 和 closable.比如我 ...
- 20180518VSTO多簿单表汇总外接程序按钮
using System; using System.Collections.Generic; using System.Linq; using System.Text; using Microsof ...
- Selenium自动化测试框架入门整理
关注嘉为科技,获取运维新知 本文主要针对Selenium自动化测试框架入门整理,只涉及总体功能及框架要点介绍说明,以及使用前提技术基础要求整理说明.作为开发人员.测试人员入门参考. 本文参考:Se ...
- android--------自定义控件 之 基本实现篇
前面简单的讲述了Android中自定义控件中的几个方法,今天通过代码来实现一个简单的案例 自定义一个扇形图 自定义控件示例: 这里先介绍继承View的方式为例 public class Circula ...
- python基础之 025 模块加载与import的使用
内容梗概: 1. 模块 2. import 3. from xxx import xxx 1.模块定义:模块就是一个包含了python定义和声明的文件,文件名就是模块的名字加上.py后缀.目前写的所有 ...
- hdu5992 kdt
题意:n个旅馆,每个有花费,m个查询,查询在某个点在c花费范围内的距离最小的旅馆 题解:kdt,建成四维,坐标两维,花费一维,id一维,实际上建树只用前两维,正常的查询,如果满足条件在更新答案即可 / ...
- shiro中JSP标签
Shiro提供了JSTL标签用于在JSP/GSP页面进行权限控制,如根据登录用户显示相应的页面按钮. 导入标签库 <%@taglib prefix="shiro" uri=& ...
- fdisk分区和挂载
查看硬盘情况 fdisk -l 修改分区类型为LVM sudo fdisk /dev/sda (/dev/sda为对应的设备名,也可能是其它名字) 按n新建分区 按p选择主分区 按t改变分区的syst ...
- js将url转换二维码
二维码生成库 qrcode.js /*from tccdn minify at 2014-6-4 14:59:43,file:/cn/c/c/qrcode.js*/ /** * @fileovervi ...





