什么是反射?
正常方法是通过一个类创建对象,反射方法就是通过一个对象找到一个类的信息。
根据类的信息来加载这个类,然后构造这个对象,然后再调用这个类中的属性和方法。(这句话是那个老师说的)
我个人理解就是,反射就是让你可以随便调用你没有实例化的东西,因为类在实例化之后就固定了,而反射就打破了这种固定模式,即使你的类没有被实例化加载到内存中,你一样可以在你要使用的时候找到它。
张孝祥说:“反射就是把Java类中的各种成分映射成相应的java类”,其细节方面就是你通过自己编写代码来将类加载到java虚拟机中,也有人称“反射”为“类的自解析”。
通过反射可以让程序变得更加灵活。
1.Class对象 VS. 实例对象
有个系统类叫Class,是所有自定义类的原型,JAVA程序运行时,每个自定义类本身也是Class类的实例化对象。
在Java中我们一般是这样使用类的:编写类,然后new对象,再调用方法。这里new出来的对象暂且称之为实例对象(instance)。其实在这之前还涉及到一个Class对象。这个Class对象就是用来创建类的所有实例对象的。
每个类都有一个Class对象。Class对象是由JVM帮我们自动生成的。每当编写并且编译了一个新类,就会产生一个Class对象(更恰当地说,是被保存在一个同名的.class文件中)。为了生成这个类的实例对象,运行这个程序的的Java虚拟机会使用被称为“类加载器”的子系统。所有类都是在对其第一次使用时,动态加载到JVM中的。类加载器会首先检查这个类的Class对象是否已经加载。若尚未加载,默认的类加载器就会根据类名查找.class文件。一旦某个类的Class对象被载入内存,它就被用来创建这个类的所有实例对象。
所有的Class对象都是java.lang.Class<T>类的实例对象。
2.为了使用一个类而做的准备工作
1.加载:这是由类加载器执行的。该步骤将查找字节码,并从这些字节码中创建一个Class对象。
2.链接:在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。
3.初始化:如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化被延迟到了对静态方法(构造器隐式地是静态的)首次引用或者对非常数静态域进行首次引用时才执行。
3.三种方法获得一个类的Class对象的引用
(1)通过java.lang.Class类的静态方法forName()
(2)调用实例对象的getClass()方法
(3)类字面常量方式(比如MyClass.class)
package com.example;class MyClass{}public class ClassTest { public static void main(String[] args) { System.out.println("第一种方式..."); Class c1 = null; try { c1 = Class.forName("com.example.MyClass"); //注意,参数是包名+类型 } catch (ClassNotFoundException e) { e.printStackTrace(); } System.out.println(c1.getName()); //输出:com.example.MyClass System.out.println(c1.getCanonicalName()); //输出:com.example.MyClass System.out.println(c1.getSimpleName()); //输出:MyClass System.out.println("第二种方式..."); Object instance = new MyClass(); Class c2 = instance.getClass(); System.out.println(c2.getName()); //输出:com.example.MyClass System.out.println("第三种方式..."); Class c3 = MyClass.class; System.out.println(c3.getName()); //输出:com.example.MyClass }}
4.泛化的Class引用
(1)Class引用总是指向某个Class对象。在声明Class引用的时候最好指定泛型。如果不能确定具体的类型,Class<?>也是优于平凡的Class的,即便它们是等价的。
(2)为了创建一个Class引用,并且它被限定为某种类型,或者该类型的任何子类型,可以将通配符与extends关键字结合,创建一个范围。如下:
Class bound = int.class;
注意:虽然Integer继承自Number,但是Integer类的Class对象和Number类的Class对象并没有半毛钱的继承关系。(也就是说Integer类的Class对象并不是Number类的Class对象的子类)
(3)还可以声明某个Class引用,它被限定为某种类型,并且是该类型的超类,可以这样:
package com.example;class BaseClass{}class ChildClass extends BaseClass{}public class ClassTest { public static void main(String[] args) { ClasschildClass = ChildClass.class; Class superClazz = childClass.getSuperclass(); System.out.println(superClazz.getName()); //输出:com.example.BaseClass }}