1.1 什么是RTTI?

维基百科的定义:In computer programming, RTTI (Run-Time Type Information, or Run-Time Type Identification) refers to a C++ mechanism that exposes information about an object's data type at runtime. Run-time type information can apply to simple data types, such as integers and characters, or to generic types. This is a C++ specialization of a more general concept called type introspection.

运行时类型信息使得你可以在程序运行时发现和使用类型信息。Runtime type information (RTTI) allows you to discover and use type information while a program is running.

简而言之:当有一个指向基类对象的引用时,使用RTTI可以查询这个引用所引用的对象的确切类型。

在java中可以通过两种方式让我们在运行时识别对象和类的信息:

(1)传统的RTTI:它假设我们在编译时已经知道了所有的类型;

(2)反射机制:它允许我们在运行时发现和使用类的信息。

1.2 为什么需要RTTI?

通常需要知道某个引用类型的变量(为了利用多态的优势,一般是泛化引用,即指向父类的引用变量)所引用的对象的具体类型, 以便根据具体的类型再做具体的不同的操作。使用RTTI可以帮助我们确定泛化引用所引用对象的确切类型。

一般是如下流程:判断引用变量的具体类型 → 转换为具体的类型 → 执行特殊类型相关的操作。

1.3 Java中类型信息在运行时是如何表示的?

这是由Class对象表示的,它包含了与类有关的信息。Java使用Class对象来执行其RTTI,即使你正在执行的是类似转换这样的操作。(Java performs its RTTI using the Class object, even if you’re doing something like a cast.)

2.1 传统的RTTI

严格的说,反射也是一种形式的RTTI,不过,一般的文档资料中把RTTI和反射分开,因为一般的,大家认为RTTI指的是传统的RTTI。

Java中使用RTTI的3种形式(java中已知的RTTI形式)

(1)传统的类型转换

比如“(Type)”,由RTTI确保类型转换的正确性,如果执行了一个错误的类型转换,会抛出ClassCastException。在Java中,所有的类型转换都是在运行才进行正确性检查的。所以在运行时需要识别一个对象的类型。注意,C++中类型转换是不会使用RTTI的。

传统的RTTI使用转型或instanceof形式(即以instanceof的形式或isInstance()的形式)实现,但都需要指定要转型的类型,比如:

    public void rtti(Object obj){
Orange orange = (Orange)obj; //其中的obj虽然是被转型了,但在编译期,就需要知道要转成的类型Orange,也就是需要Orange的.class文件
//Class clazz = Class.forName("rtti.Orange"); //相对的,反射完全在运行时在通过Class类来确定类型,不需要提前加载Orange的.class文件。
}

(2)代表对象类型的Class对象

通过查询Class对象可以获取运行时所需的信息。(用了Class对象,不代表就是反射,如果只是用Class对象cast成指定的类,那就还是传统的RTTI)。其实反射和RTTI并没有什么本质的区别,因为java的类都是在运行是加载并解析的,而且两者通过Class对象来获取类型信息。只是RTTI能够维护的类型都是编译时已知的类型,而反射可以使用一些在编译时完全不可知的类型。

        Class clazz = Orange.class;    //获取Class对象,Orange这个类必须是编译时可知的
System.out.println(clazz.getName()); //利用class对应获取各种类型信息
System.out.println(clazz.getTypeName());
System.out.println(clazz.getSuperclass());
System.out.println(clazz.isInterface());

(3)instanceof或Class.isInstance()

传统的RTTI使用转型或instanceof形式(即以instanceof的形式或isInstance()的形式)实现,但都需要指定要转型的类型,比如:

        if(x instanceof Apple){    //Apple这个类必须是编译时可知的,编译器在编译时会去打开和检查Apple对应的.class文件
((Apple)x).info(); } x.getClass().isInstance(apple);//声明apple对象的类必须是编译时可知的

2.2 RTTI与反射的区别

使用RTTI必须满足一个条件:如果想知道泛化引用的确切类型,那么这个类在编译时必须可知。

通过RTTI可以查询到某个对象的确切类型,这个类型在编译时必须已知(要么是系统类库中的类,要么是你自己在代码中定义的类,因为一个Class定义最终会转换成一个.class文件,所以编译器会去查找看有没有对应的.class文件,没有的话在编译时就报错“错误:找不到符号”,这个过程是在编译阶段完成的),这样才能使用RTTI识别它。换句话说,在编译时,编译器必须知道所有要通过RTTI来处理的类。

RTTI与反射的区别如下:

RTTI:由编译器在编译时打开和检查*.class文件。

反射机制:由JVM在运行时打开和检查*.class文件。

            Class clazz1 = Class.forName("rtti.Apple");    //在编译阶段编译器不会检查和打开字符串"rtti.Apple"指定的类对应的.class文件,而是在运行期间由JVM加载。
System.out.println(clazz1.getName());
        Fruit fruit = new Bnana();    //编译阶段编译器会去查找Fruit类和Apple类各自对应的.class文件。
Class clazz2 = fruit.getClass();
System.out.println(clazz2.getName());

疑问:编译器在编译时打开和检查*.class文件的这一过程中具体都干了什么?

3.1 反射机制

维基百科的定义:In computer science, reflection is the ability of a computer program to examine (see type introspection) and modify the structure and behavior (specifically the values, meta-data, properties and functions) of the program at runtime.

并不是所有的类型都能在编译的时候确定下来(比如有输入一个类名显示出这个类的所有方法的程序),所以得把打开和检查.class文件的工作推到运行时再处理。典型的场景是“基于构件的编程”和远程方法调用(RMI) 。

Class.forName()生成的结果在编译时是不可知的,也就是说,编译时不会去验证参数字符串对应的类是否存在。这都是在运行的时候由JVM处理的,所以这个方法会抛出ClassNotFoundException。

Class类支持反射,是在java.lang.reflect中包含了Field/Method/Constructor类,每个类都实现了Member接口。这些类型的对象都是由JVM在运行时创建的,用来表示未知类里对应的成员。如可以用Constructor类创建新的对象,用get()和set()方法读取和修改与Field对象关联的字段,用invoke()方法调用与Method对象关联的方法。同时,还可以调用getFields()、getMethods()、getConstructors()等方法来返回表示字段、方法以及构造器的对象数组。这样,未知的对象的类信息在运行时就能被完全确定下来,而在编译时不需要知道任何信息。

3.2 如何获取Class对象

(1)Class.forName(String packageNamePlusClassName);

(2)Object.getClass();

(3)类字面常量

3.3泛化的Class引用

(1)通配符方式

Class<?> intClass  = int.class;

(2)extends实现范围限定

Class<? extends NUmber> bounded = int.class;

(3)super方式

Class<Apple> apClass = Apple.class;
Apple apple = apClass.newInstance();
Class<? super Fruit> up = apClass.getSuperclass();

3.4 instanceof与class的等价性

(1)instanceof 与 isInstance()生成的结果完全一样,equal与==也一样。

(2)Instanceof与isInstance()保持了类型的概念,它指的是“你是这个类,或者是这个类的派生类吗?”

(3)equal与==比较实际的Class对象,没有考虑继承——它或者是这个确切的类型,或者不是。

4.总结

JAVA编程思想,第14章末尾的总结一节的概括:

第1段的意思,使用RTTI可以发现引用变量的确切类型,然后根据引用变量具体的确切类型执行对应的操作(类似switch语句,P327的例子)。这种方式就没有使用多态,失去了多态的价值。所以尽量使用多态机制,只是在必需的时候才使用RTTI。

第2-4段,概述了RTTI在部分场景下的优势,分别举例说明使用RTTI可以解决的问题。

参考资料

JAVA编程思想,第14章,第8章 8.5.2

http://www.cnblogs.com/zhguang/p/3091378.htm

http://blog.csdn.net/cannel_2020/article/details/7226108

http://blog.csdn.net/a81895898/article/details/8457623

http://blog.sina.com.cn/s/blog_548c8a8301013j6u.html

Java RTTI机制与反射机制的更多相关文章

  1. Java进阶之reflection(反射机制)——反射概念与基础

    反射机制是Java动态性之一,而说到动态性首先得了解动态语言.那么何为动态语言? 一.动态语言 动态语言,是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化.比如常见 ...

  2. java的泛型与反射机制

    什么是泛型? 泛型,即“参数化类型”.顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参) ...

  3. Java 核心类库之反射机制

    1:什么是反射机制? 2:反射机制它可以做什么呢? 3:反射机制对应的API又是什么? 1):通过反射机制来获取一个对象的全限定名称(完整包名),和类名: 2):实例化Class对象 3):获取对象的 ...

  4. java.lang.Class<T> -- 反射机制

    1反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为jav ...

  5. 在JAVA中,关于反射机制的讨论

    一.什么是反射机制         简单的来说,反射机制指的是程序在运行时能够获取自身的信息.在java中,只要给定类的名字,     那么就可以通过反射机制来获得类的所有信息. 二.哪里用到反射机制 ...

  6. java工厂类与反射机制

    java 简单工厂类 2012-04-22 15:44:07|  分类: java |  标签:java工厂类  简单工厂类  |举报|字号 订阅     简单工厂模式需要由以下角色组成: 接口    ...

  7. Java SE之初探反射机制

    [Keywords]:Java,Hibernate,虚拟机,框架,SQL [Abstract]:   反射的概念:所谓的反射就是java语言在运行时拥有一项自观的能力,反射使您的程序代码能够得到装载到 ...

  8. JAVA(五)反射机制/Annotation

    成鹏致远 | lcw.cnblog.com |2014-02-04 反射机制 1.认识Class类 在正常情况下,必须知道一个类的完整路径之后才可以实例化对象,但是在 java中也允许通过一个对象来找 ...

  9. java 类加载机制和反射机制

    一.类的加载机制 jvm把class文件加载到内存,并对数据进行校验.解析和初始化,最终形成jvm可以直接使用的java类型的过程.(1)加载         将class文件字节码内容加载到内存中, ...

随机推荐

  1. sg函数与博弈论2

    参考链接: http://blog.sina.com.cn/s/blog_51cea4040100h3l9.html 这篇主要就是讲anti-sg.multi-sg和every-sg的. 例1 poj ...

  2. 用Myisamchk让MySQL数据表更健康

    用Myisamchk让MySQL数据表更健康 2011-03-15 09:15 水太深 ITPUB 字号:T | T 为了让MySQL数据库中的数据表“更健康”,就需要对其进行定期体检.在这里笔者推荐 ...

  3. 【转】CSS样式覆盖规则

    大家都知道CSS的全称叫做“层叠样式表”,但估计很多人都不知道“层叠”二字的含义.其实,“层叠”指的就是样式的覆盖,当一个元素被运用上多种样式,并且出现重名的样式属性时,浏览器必须从中选择一个属性值, ...

  4. 如何获取Flickr图片链接地址作为外链图片

    Flickr,雅虎旗下图片分享网站.为一家提供免费及付费数位照片储存.分享方案之线上服务,也提供网络社群服务的平台.其重要特点就是基于社会网络的人际关系的拓展与内容的组织.这个网站的功能之强大,已超出 ...

  5. js基础知识温习:js中的对象

    在JavaScript中对象是一个无序属性的集合,其属性可以包含基本值.对象或者函数. 对象最简单的创建方式 JavaScript中创建对象最简单的方式就是创建一个Object对象的实例,然后再添加属 ...

  6. [BZOJ 1997][HNOI2010]Planar(2-SAT)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1997 分析: 考虑每条边是在圈子里面还是圈子外面 所以就变成了2-SAT判定问题了= ...

  7. C语言字符串与数字相互转换

    在C/C++语言中没有专门的字符串变量,通常用字符数组来存放字符串.字符串是以“\0”作为结束符.C/C++提供了丰富的字符串处理函数,下面列出了几个最常用的函数. ● 字符串输出函数puts. ● ...

  8. 样条函数 -- spline function

    一类分段(片)光滑.并且在各段交接处也有一定光滑性的函数.简称样条.样条一词来源于工程绘图人员为了将一些指定点连接成一条光顺曲线所使用的工具,即富有弹性的细木条或薄钢条.由这样的样条形成的曲线在连接点 ...

  9. android之广播(二)

    广播接受者不仅可以通过清单文件来向系统注册,也可以通过代码来注册.并且有的广播必须通过代码来注册广播接受者. 锁屏和解锁广播 电量改变广播 打开屏幕和关闭屏幕 这里将广播接收者写在服务里面 <? ...

  10. 【JavaEE企业应用实战学习记录】servlet3.0上传文件

    <%-- Created by IntelliJ IDEA. User: Administrator Date: 2016/10/6 Time: 14:20 To change this tem ...