0.参考笔记
1.为什么需要RTTi
RTTI(Run-Time Type Identification):在运行时识别一个对象类型
JAVA在运行时 有时要 识别对象和类的信息这个机制叫RTTI。Java提供了两种机制去做这件事。传统的RTTI 和 反射。
- 传统的RTTI 假定编译时就已经知道了所有的类型。
- 反射 允许在运行时发现和使用类型信息
传统的RTTI 在编译期通过Class文件识别类型信息,反射在运行期通过Class文件识别类型信息。
Java类型转换都发生在运行时期。
2.Class对象
Class对象是由Class这个类产生的,它包含了与某个类有关的信息。
每个类都有Class对象,即编译(java先编译形成中间码)了一个新类就会产生一个该类的Class对象,并且保存在.class文件中。Class对象就是用来产生“常规”对象的。
Java使用Class对象来执行RTTI。
所有类都是第一次使用时动态加载到jvm中。 动态加载就是需要时再加载不使用不加载。
只要创建了对一个类的静态成员的引用就会加载该类。new 的时候加载类说明 类的构造器虽然没写static但也是静态方法。
一个类的Class对象载入内存后,他就会用来创建该类的对象。
获得Class对象
Class类的静态方法forName()可以通过传入一个全限定类名(包含包名)返回一个该类的Class类对象引用,此时该类会被加载到内存。
Class.forName(“thinking14class.Test”)
运行时要获得一个类的信息(类型信息)可以通过一个该类的Class对象获得,使用Class.forName()就可以做到,而不必持有该类型的对象通过该对象获得。
如果有了一个实例对象可以调用getClass()方法来获得Class对象。
14.2.1 类字面常量
Java还提供了另一种方法来生成对Class对象的引用,即使用类字面常量。
例如类为:FancyToy
那么引用为:FacyToy.class
这样做不仅更简单,而且更安全,因为它在编译时就会受到检查(因此不需要置于try语句块中)。并且它根除了对forName()方法的调用,所以也更高效。
类字面变量不仅可以应用于普通的类,也可以应用于接口、数组以及基本数据类型。另外,对于基本数据类型的包装器类,还有一个标准字段TYPE。TYPE字段是一个引用,指向对应的基本数据类型的Class对象。
boolean.class 等价于 Boolean.TYPE
char.class 等价于 Character.TYPE
byte.class 等价于 Byte.TYPE
…
使用“.class”来创建对Class对象的引用时,不会自动地初始化该Class对象。为了使用类而做的准备工作实际包含三个步骤:
-
加载,这是由类加载器执行的。该步骤将查找字节码(通常在classpath所指定的路径汇总查找,但这并非是必须得),并从这些字节码中创建一个Class对象。
-
链接。在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必须得话,将解析这个类创建的对其他类的所有引用。
-
初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
3.类型转换前先做检查
传统的类型转换由RTTI确保正确性。
instanceof关键字(二元操作符) ,返回一个Boolean值,告诉我们对象是不是某个类或该类派生类的实例,他判断的是类型。
if (a instanceof A) //对象a是不是属于A类型,A可能是a父类的父类,如果是这样也返回true
instanceof 不能比较Class对象,对于Class对象使用isAssignableFrom()判断
if (as.isAssignableFrom(cs))// Class对象cs所在类是不是属于Class对象as所在类或者派生类
动态的instanceof :Class对象的isInstance(Object o)方法判断该Class对象是不是o类的(如果o是class对象所在类则返回true,否则返回false哪怕o是所在类的父类)。
if (cs.isInstance©) //如果c是class对象所在类则返回true,否则返回false,哪怕c是所在类的父类
4.注册工厂
工厂设计模式
5.instanceof和Class的等价性
isntanceof和isInstance生成的结果完全相同
equals和==也完全一致
instanceof保持了类型的概念:指的是“你是这个类或者派生类吗”
而如果用= =比较的就没考虑继承 指的是是否为确切类型
6.反射
可以通过反射获取到一个类的所有属性和方法(包括私有类型)
参考笔记
8.空对象
参考
1、定义
空对象就是一个正常的对象,只不过你用这个对象来替代null。让一个类实现Null接口。
2、作用
当使用null表示缺少对象时,在每次引用前都要测试其是否为null,因此需要在代码中加入判断语句,当判断语句变多时,代码就变得杂乱,使用空对象可以减少判断的语句。
例子:
假如有一个Person类,Person类有姓名地址等属性。还有一个Position(职位)类,职位类包含了职位名称和担任职位的人(一个Person对象)等属性。
一般情况下是先确定有哪些职位然后再确定职位由谁担任,那么在没有确定担任人的时候,可以将Position类中的Person对象设为null,但是也可以将其设为一个空对象。
9.接口与类型信息
接口并不能保护客户端不调用实现类中其他的方法,比如通过反射。
哪怕是private的方法或字段,都可以使用Method.setAccessible(true)之后访问。