运行时类型信息(RunTime Type Information,RTTI)使得你在程序运行时发现和使用类型
信息。RTTI主要用来运行时获取向上转型之后的对象到底是什么具体的类型。
1.Class对象:
JAVA使用Class对象来执行RTTI。每个类都有一个Class对象,它用来创建这个类的所有
对象,反过来说,每个类的所有对象都会关联同一个Class对象(对于数组来说,维数、类
型一致的数组的Class 对象才是相同的),每个对象的创建都依赖于Class对象的是否创建,
Class对象的创建发生在类加载(java.lang.ClassLoader)的时候。
java.lang.Class类实现了Serializable、GenericDeclaration、Type、AnnotatedElement四个接口,
分别实现了可序列化、泛型定义、类型、元数据(注解)的功能。
你可以把Class对象理解为一个类在内存中的接口代理(它代理了这个类的类型信息、方法
签名、属性),JVM加载一个类的时候首先创建Class对象,然后创建这个类的每个实例的
时候都使用这个Class 对象。
Class只有一个私有的无参构造方法,也就是说Class的对象创建只有JVM可以完成。
如何验证同一个类的多个对象的Class对象是一个呢?
Cf1 cf1 = new Cf1();
Class clazz = Cf1.class;
System.out.println(cf1.getClass() == clazz);
我们知道==用来比较引用是否相等(也就是同一个引用),上面的输出语句结果是true。那
么Class对象是否相等是JAVA对象中唯一可以使用==判断的。
如何获取Class对象:
1.所有的引用数据类型(类-类型)的类名、基本数据类型都可以通过.class方式获取其Class
对象(对于基本数据类型的封装类还可以通过.TYPE 的方式获取其Class 对象,但要注
意.TYPE 实际上获取的封装类对应的基本类型的Class 对象的引用,那么你可以判断出
int.class==Integer.TYPE 返回true,int.class==Integer.class 返回false!),通过这种方式
不会初始化静态域,使用.class、.TYPE 的方式获取Class对象叫做类的字面常量;
2.Class 的forName(String name)传入一个类的完整类路径也可以获得Class 对象,但由于使
用的是字符串,必须强制转换才可以获取泛型的Class<T>的Class对象,并且你必须获取这
个方法可能抛出的ClassNotFoundException异常。
2.对于引用数据类的引用(必须初始化),可以通过Object 类继承的getClass()方法获取这
个引用的Class对象,由于引用已经被初始化,所以这种方式也不会初始化静态域,因为静
态域已经被初始化过。另外,前面两种方式如果说是创建Class对象,那么这种方式应该是
取得Class对象,因为类的实例已经被创建,那么Class对象也一定早就被创建。
Class的常用方法:
l forName(String name):这是一个静态方法,传入的参数是一个类的完整类路径的
字符串,返回这个类的Class 对象,前面说过Class 对象的创建发生在类的加载时,所
以这个方法会导致静态成员被调用;
l forName(String name,boolean initialize,ClassLoader loader):这是上面的方
法的重载方法,initialize指定在创建Class对象时是否初始化这个类(即是否执行静态成
员,由于在一次JVM的执行中,静态成员的初始化只类加载的时候执行一次,所以如果
之前这个类已经被加载,那么即使initialize为true也不会再次执行静态成员的加载),
loader指定使用哪个类加载器的实现类
(Thread.currentThread().getContextClassLoader()可以获取当前线程使用的类
加载器)。forName(***)方法不可以获取基本数据类型的Class对象。
如果要测试initialize是否起作用,请不要在main()方法测试自身类,因为main()是静态方法,
执行这个方法会导致静态域被初始化,所以你的initialize无论是true还是false,效果都是一样
的。
l asSubClass(Class superClass):这个方法是将父类的class 对象作为参数传入,并
将其强制转换成当前的Class对象(子类的Class对象)。
例:
Class<List> clazz = List.class;
Class<? extends List> subClazz = ArrayList.class.asSubclass(clazz);
System.out.println(subClazz.getCanonicalName());
注意红色的部分不能写成Class<ArrayList>形式,因为asSubclass()只知道是要转换成子
类的Class对象,但不知道是哪个子类。
l cast(Object o):这个方法是将传入的对象强制转换成Class 对象所代表的类型的对
象;
信息。RTTI主要用来运行时获取向上转型之后的对象到底是什么具体的类型。
1.Class对象:
JAVA使用Class对象来执行RTTI。每个类都有一个Class对象,它用来创建这个类的所有
对象,反过来说,每个类的所有对象都会关联同一个Class对象(对于数组来说,维数、类
型一致的数组的Class 对象才是相同的),每个对象的创建都依赖于Class对象的是否创建,
Class对象的创建发生在类加载(java.lang.ClassLoader)的时候。
java.lang.Class类实现了Serializable、GenericDeclaration、Type、AnnotatedElement四个接口,
分别实现了可序列化、泛型定义、类型、元数据(注解)的功能。
你可以把Class对象理解为一个类在内存中的接口代理(它代理了这个类的类型信息、方法
签名、属性),JVM加载一个类的时候首先创建Class对象,然后创建这个类的每个实例的
时候都使用这个Class 对象。
Class只有一个私有的无参构造方法,也就是说Class的对象创建只有JVM可以完成。
如何验证同一个类的多个对象的Class对象是一个呢?
Cf1 cf1 = new Cf1();
Class clazz = Cf1.class;
System.out.println(cf1.getClass() == clazz);
我们知道==用来比较引用是否相等(也就是同一个引用),上面的输出语句结果是true。那
么Class对象是否相等是JAVA对象中唯一可以使用==判断的。
如何获取Class对象:
1.所有的引用数据类型(类-类型)的类名、基本数据类型都可以通过.class方式获取其Class
对象(对于基本数据类型的封装类还可以通过.TYPE 的方式获取其Class 对象,但要注
意.TYPE 实际上获取的封装类对应的基本类型的Class 对象的引用,那么你可以判断出
int.class==Integer.TYPE 返回true,int.class==Integer.class 返回false!),通过这种方式
不会初始化静态域,使用.class、.TYPE 的方式获取Class对象叫做类的字面常量;
2.Class 的forName(String name)传入一个类的完整类路径也可以获得Class 对象,但由于使
用的是字符串,必须强制转换才可以获取泛型的Class<T>的Class对象,并且你必须获取这
个方法可能抛出的ClassNotFoundException异常。
2.对于引用数据类的引用(必须初始化),可以通过Object 类继承的getClass()方法获取这
个引用的Class对象,由于引用已经被初始化,所以这种方式也不会初始化静态域,因为静
态域已经被初始化过。另外,前面两种方式如果说是创建Class对象,那么这种方式应该是
取得Class对象,因为类的实例已经被创建,那么Class对象也一定早就被创建。
Class的常用方法:
l forName(String name):这是一个静态方法,传入的参数是一个类的完整类路径的
字符串,返回这个类的Class 对象,前面说过Class 对象的创建发生在类的加载时,所
以这个方法会导致静态成员被调用;
l forName(String name,boolean initialize,ClassLoader loader):这是上面的方
法的重载方法,initialize指定在创建Class对象时是否初始化这个类(即是否执行静态成
员,由于在一次JVM的执行中,静态成员的初始化只类加载的时候执行一次,所以如果
之前这个类已经被加载,那么即使initialize为true也不会再次执行静态成员的加载),
loader指定使用哪个类加载器的实现类
(Thread.currentThread().getContextClassLoader()可以获取当前线程使用的类
加载器)。forName(***)方法不可以获取基本数据类型的Class对象。
如果要测试initialize是否起作用,请不要在main()方法测试自身类,因为main()是静态方法,
执行这个方法会导致静态域被初始化,所以你的initialize无论是true还是false,效果都是一样
的。
l asSubClass(Class superClass):这个方法是将父类的class 对象作为参数传入,并
将其强制转换成当前的Class对象(子类的Class对象)。
例:
Class<List> clazz = List.class;
Class<? extends List> subClazz = ArrayList.class.asSubclass(clazz);
System.out.println(subClazz.getCanonicalName());
注意红色的部分不能写成Class<ArrayList>形式,因为asSubclass()只知道是要转换成子
类的Class对象,但不知道是哪个子类。
l cast(Object o):这个方法是将传入的对象强制转换成Class 对象所代表的类型的对
象;