C++中的类型识别
1,为什么会提出类型识别概念呢?
1,为什么在 C 语言中没有提出这个概念呢,就是因为在 C++ 中引入了面向对象的特性,面向对象里面有一个非常重要的原则就是赋值兼容性原则;
2,在面向对象中可能出现下面的情况:
1,基类指针指向子类对象;
2,基类引用成为子类对象的别名;
3,示意图:

1,p 指针字面意思是用来指向父类对象,而 p 又指向了子类对象;
2,当得到 p 指针后,我们是否有能力判断当前指针指向的是父类对象还是子类对象吗?目前没有办法通过一个父类指针来判断它指向的究竟是父类对象还是一个子类对象;
3,上图中展示的指针 p “静态类型”是 Base*,“静态类型”是指针它期望的类型,赋值兼容性使得 p 有可能指向的不是它期望的类型,而是子类类型,这时指针所指向的类型就叫做“动态类型”;
4,引用 r 静态类型是 Base&,r 的本意是引用父类的别名,但由于赋值兼容性,它也可以引用子类的别名,无法确定它的别名究竟是父类别名还是子类别名;
3,类型识别:
1,静态类型:
1,变量(对象)自身的类型;
1,写代码时就可以完全确定;
2,动态类型:
1,指针(引用)所指向对象的实际类型;
1,写代码时确定不了;

1,b 在代码运行时不一定指向父类对象,它有可能指向一个子类对象,因此 b 指针指向的是动态类型;
2,当 b 指针指向的是子类对象,则程序正常运行,将父类类型转换为子类类型是非常危险的,有可能产生 Bug;
(3),基类指针是否可以强制类型转换为子类指针取决于动态类型;
1,这样的转换和静态类型转换没有丝毫关系;
3,C++ 中如何得到动态类型?
4,动态类型识别:
1,解决方案(老师的思路):利用多态
1,在基类中定义虚函数返回具体的类型信息;
2,所有的派生类都必须实现类型相关的虚函数;
3,每个类中的类型虚函数都需要不同的实现;
1,调用类型虚函数就可以知道当前的对象究竟是什么类型,这样就可 以得到动态类型,达到动态类型识别效果;
5,动态类型识别编程实验:
#include <iostream>
#include <string> using namespace std; class Base
{
public:
virtual string type()
{
return "Base";
}
}; class Derived : public Base
{
public:
string type()
{
return "Derived";
} void printf()
{
cout << "I'm a Derived." << endl;
}
}; class Child : public Base
{
public:
string type()
{
return "Child";
}
}; void test(Base* b)
{
/* 危险的转换方式 */
// Derived* d = static_cast<Derived*>(b); if( b->type() == "Derived" )
{
Derived* d = static_cast<Derived*>(b); // 转换 d->printf(); // 使用指针 d,这里只是说明问题;
} // cout << dynamic_cast<Derived*>(b) << endl; // 不够用,舍弃;
} int main(int argc, char *argv[])
{
Base b;
Derived d;
Child c; test(&b); // 用 dynamic_cast 时,父类转换子类不成功,返回 0;
test(&d);
test(&c); // 用 dynamic_cast 时,子类间转换不成功,返回 0; return ;
}
1,最正规的转换用法应该是 dynamic_cast,可以判断 dynamic_cast 返回值来判断指针转换是否成功;
2,这里我们需要知道动态的类型究竟是什么,所以 dynamic_cast 在这里根本不够用,仅仅只能告诉我们转换是否成功;
3,这里通过虚函数返回类型名的方式就可以成功的得到动态类型名;
4,但是这里有个问题是长期维护,只要我们写一个新的类出来,必须都要实现这个虚函数,如果没有实现这个虚函数,后果将会造成对象的动态类型和实 际不符合的错误,并且造成 if() 语句中的调用不会成功;
6,多态解决方案的缺陷:
1,必须从基类开始提供类型虚函数;
2,所有的派生类都必须重写类型虚函数;
1,长期维护并不好;
3,每个派生类的类型名必须唯一;
7,C++ 提供了 typeid 关键字用于获取类型信息:
1,typeid 关键字返回对应参数的类型信息;
1,类的和基础类型的类型信息;
2,typeid 返回一个 type_info 类对象;
3,当 typeid 的参数为 NULL 时将抛出异常;
1,参数可以是类型名、变量名;
8,typeid 关键字的使用:
1,代码示例:
int i = ; const type_info& tiv = typeid(i); // 将 i 的类型信息放到 type_info 中去;
const type_info& tii = typeid(int); cout << (tiv == tii) << endl;
9,typeid 的注意事项:
1,当参数为类型时,返回静态类型信息;
2,当参数为变量时:
1,参数变量内部不存在虚函数表时,返回静态类型信息;
2,参数变量内部存在虚函数表时,返回动态类型信息;
10,tpeid 类型识别编程实验:
#include <iostream>
#include <string>
#include <typeinfo> // typeid 返回的对象类型 tyep_info 所对应的头文件; using namespace std; class Base
{
public:
virtual ~Base()
{
}
}; class Derived : public Base
{
public:
void printf()
{
cout << "I'm a Derived." << endl;
}
}; void test(Base* b)
{
const type_info& tb = typeid(*b); cout << tb.name() << endl;
} int main(int argc, char *argv[])
{
int i = ; const type_info& tiv = typeid(i);
const type_info& tii = typeid(int); cout << (tiv == tii) << endl; // 1; Base b;
Derived d; /* 通过不同的对象调用得到类型信息相同,因为 b 对象没有虚函数表,此时返回静态类型信息 */
test(&b); // 4Base;
test(&d); // 4Base; /* 对象 b 里加上虚函数表后,返回动态类型信息 */
test(&b); // 4Base;(在 Linux 的 g++ 编译器下面显示的)Base;
//(在 windows 的 BCC 编译器下面显示的)
test(&d); // 7Derived;(在 Linux 的 g++ 编译器下面显示的)
// Derived;(在 windows 的 BCC 编译器下面显示的) return ;
}
1,typeid 在不同的编译器内部实现是不同的;
11,小结:
1,C++ 中有静态类型和动态类型的概念;
2,利用多态能够实现对象的动态类型识别;
1,维护成本高,一旦不小心出错,整个项目就会有 bug;
3,typeid 是专用于类型识别的关键字;
4,typeid 能够返回对象的动态类型信息;
1,使用 typeid 的时候,一定不要进行某些类型上面的假设,因为不同的编译器处理类型的名字它的方式是不一样的;
C++中的类型识别的更多相关文章
- 第66课 C++中的类型识别
1. 类型识别 (1)在面向对象中可能出现下面的情况 ①基类指针指向子类对象 ②基类引用成为子类对象的别名 ▲静态类型——变量(对象)自身的类型(定义变量类型时类型或参数类型) ▲动态类型——指针(引 ...
- JS中的类型识别
JS为弱类型语言,所以类型识别对JS而言尤为重要,JS中常用的类型识别方法有4种:typeof.Object.prototype.toString.constructor和instanceof. (1 ...
- 框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解)
框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解) 一丶什么是RTTI,以及RTTI怎么设计 通过第一讲,我们知道了怎么样升成一个窗口了,以及简单的消息循环. 第二讲则是主要讲解RTTI ...
- 关于js中的类型内容总结(类型识别)
JS 有7种数据类型: 6种原始类型:Boollean String Number Null Underfined Symbol 引用类型:Object 类型识别主要有以下四 ...
- RTTI 运行时类型识别 及异常处理
RTTI 运行时类型识别 typeid ------ dynamic_cast dynamic_cast 注意事项: 1.只能应用于指针和引用之间的转化 2.要转换的类型中必须包含虚函数 3. ...
- RTTI (Run-Time Type Identification,通过运行时类型识别) 转
参考一: RTTI(Run-Time Type Identification,通过运行时类型识别)程序能够使用基类的指针或引用来检查这些指针或引用所指的对象的实际派生类型. RTTI提供了以下两个 ...
- 数据库中字段类型对应的C#中的数据类型
数据库中字段类型对应C#中的数据类型: 数据库 C#程序 int int32 text string bigint int64 binary System.Byte[] ...
- MFC六大核心机制之二:运行时类型识别(RTTI)
上一节讲的是MFC六大核心机制之一:MFC程序的初始化,本节继续讲解MFC六大核心机制之二:运行时类型识别(RTTI). typeid运算子 运行时类型识别(RTTI)即是程序执行过程中知道某个对象属 ...
- 拼接json时小心C#中bool类型转化
C#中bool类型的值,在ToString时会有如下转化:true—>Ture ; false—>False这是拼接到json串中就会出现如下结果:{ "no": &q ...
随机推荐
- 51 Nod N的阶乘的长度 (斯特林近似)
1058 N的阶乘的长度 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 输入N求N的阶乘的10进制表示的长度.例如6! = 720,长度为3. Inp ...
- Android 属性动画监听事件与一个菜单的例子
简单监听事件 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 3 ...
- 关于多个py文件生成一个可运行exe文件(用pyinstaller)
首先下载Pyinstaller,在cmd命令下执行:pip installer Pyinstaller,不需要关心安装在哪 然后把所有相关的py文件都放在一个目录下 在那个目录下执行cmd命令:pyi ...
- xwiki使用中的问题
xwiki 内存限制 问题重现: xwiki启动后内存.cpu一直上涨,不回落,启动后服务访问速度越来越慢,最后无法访问 分析: xwiki在启动时会消耗大量内存和cpu,增加tomcat最大内存限制 ...
- JQuery实现表格动态增加行并对新行添加事件
实现功能: 通常在编辑表格时表格的行数是不确定的,如果一次增加太多行可能导致页面内容太多,反应变慢:通过此程序实现表格动态增加行,一直保持最下面有多个空白行. 效果: 一:原始页面 二:表1增加新行并 ...
- CoreData编辑器
如何你开发iOS使用的是CoreData数据库的话,肯定想要一个可以查看和编辑CoreData数据库的工具,今天给大家推荐一个工具Core-Data-Editor 下载地址:https://githu ...
- Docker入门-介绍和安装
Docker容器 Docker是什么 Docker最初是dotCloud公司创建人Solomon Hykes在法国期间发起的一个公司内部项目,它是基于dotCloud公司多年云服务技术的一次革新,并于 ...
- Spring之AOP配置
特别提示:本人博客部分有参考网络其他博客,但均是本人亲手编写过并验证通过.如发现博客有错误,请及时提出以免误导其他人,谢谢!欢迎转载,但记得标明文章出处:http://www.cnblogs.com/ ...
- ES6 字符串的扩展(待细读)
1.确定字符串中是否含有某个字符串 indexof(value,num):可返回某个指定的字符串值在字符串中首次出现的位置.ES5方法,num范围(0~length-1) includes(value ...
- 万变的Web,不变的CRUD
用JSP+Servlet写程序,到Struts,Spring,hibernate写程序,到现在Spring Cloud分布式写程序,到底有多大区别,是不是还在写CRUD? 看着JD上各种要求,简直是S ...