要说Java反射,首先说一下动态语言,了解了动态语言的特性方便我们更好的理解反射。
动态语言
动态语言是指在运行期间内可以改变其结构的语言,例如新的函数可以被引进,已有的函数可以在结构上发生变化。像典型的C#、JavaScript、Python等。
优点:方便阅读,清晰明了。
缺点:不方便调试。
Java并不是动态语言,但有了反射机制,Java可以称之为半动态语言。
反射
在 Java 中的反射机制是指在运行状态中,对于任意一个类都能够知道这个类所有的属性和方法;并且对于任意一个对象,都能够调用它的任意一个方法;这种动态获取信息以及动态调用对象方法的功能成为 Java 语言的反射机制。
反射的应用场景
在 Java 程序中许多对象在运行是都会出现两种类型:编译时类型和运行时类型。 编译时的类型由声明对象时实用的类型来决定,运行时的类型由实际赋值给对象的类型决定 。
Person person = new Student();
//其中编译时类型为 Person,运行时类型为 Student。
程序在运行时还可能接收到外部传入的对象,该对象的编译时类型为 Object,但是程序有需要调用该对象的运行时类型的方法。为了解决这些问题,程序需要在运行时发现对象和类的真实信息。然而,如果编译时根本无法预知该对象和类属于哪些类,程序只能依靠运行时信息来发现该对象和类的真实信息,此时就必须使用到反射了。
Java反射API
反射 API 用来生成 JVM 中的类、接口或则对象的信息。
- Class 类:反射的核心类,可以获取类的属性,方法等信息。
- Field 类:Java.lang.reflec 包中的类,表示类的成员变量,可以用来获取和设置类之中的属性
值。 - Method 类: Java.lang.reflec 包中的类,表示类的方法,它可以用来获取类中的方法信息或
者执行方法。 - Constructor 类: Java.lang.reflec 包中的类,表示类的构造方法。
反射的使用步骤
调用某个对象的 getClass()方法
调用某个类的 class 属性来获取该类对应的 Class 对象
使用 Class 类中的 forName()静态方法(最安全/性能最好)
public class Person {
public String name;
private int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public void eat(){
System.out.println("this is public method!");
}
private void dance(){
System.out.println("this is private method!");
}
}
通过反射获取类的信息
public class Test {
@org.junit.Test
public void test1() throws ClassNotFoundException {
Person p = new Person();
//获取类信息
//调用某个对象的 getClass()方法
Class clazz = p.getClass();
//调用某个类的 class 属性来获取该类对应的 Class 对象
Class clazz1 = Person.class;
//使用 Class 类中的 forName()静态方法(最安全/性能最好)
Class clazz2 = Class.forName("com.xzy.reflect.Person");
System.out.println(clazz);
System.out.println(clazz1);
System.out.println(clazz2);
System.out.println(clazz == clazz1);
System.out.println(clazz == clazz2);
}
}
运行结果:
class com.xzy.reflect.Person
class com.xzy.reflect.Person
class com.xzy.reflect.Person
true
true
当我们获得了想要操作的类的 Class 对象后,可以通过 Class 类中的方法获取并查看该类中的方法和属性。
public class Test {
@org.junit.Test
public void test2() throws ClassNotFoundException {
Class clazz = Class.forName("com.xzy.reflect.Person");
//获取Person类的所有方法
Method[] declaredMethods = clazz.getDeclaredMethods();
System.out.println("获取到的所有方法:");
for (Method methods : declaredMethods) {
System.out.println(methods.toString());
}
//获取Person类的所有成员属性
System.out.println("获取到的所有属性:");
Field[] field = clazz.getDeclaredFields();
for (Field f : field) {
System.out.println(f.toString());
}
//获取 Person 类的所有构造方法信息
Constructor[] constructor = clazz.getDeclaredConstructors();
System.out.println("获取到的所有构造器:");
for (Constructor c : constructor) {
System.out.println(c.toString());
}
}
}
运行结果:
获取到的所有方法:
public void com.xzy.reflect.Person.eat()
private void com.xzy.reflect.Person.dance()
获取到的所有属性:
public java.lang.String com.xzy.reflect.Person.name
private int com.xzy.reflect.Person.age
获取到的所有构造器:
public com.xzy.reflect.Person()
public com.xzy.reflect.Person(java.lang.String,int)
Process finished with exit code 0
创建对象的两种方法
Class 对象的 newInstance()
- 使用 Class 对象的 newInstance()方法来创建该 Class 对象对应类的实例,但是这种方法要求该 Class 对象对应的类有默认的空构造器。调用 Constructor对象的 newInstance()
- 先使用 Class 对象获取指定的 Constructor 对象,再调用 Constructor 对象的 newInstance()方法来创建 Class 对象对应类的实例,通过这种方法可以选定构造方法创建实例。
@org.junit.Test
public void test3() throws Exception {
//获取 Person 类的 Class 对象
Class clazz = Class.forName("com.xzy.reflect.Person");
//使用.newInstance 方法创建对象
Person p = (Person) clazz.newInstance();
//获取构造方法并创建对象
Constructor c = clazz.getDeclaredConstructor(String.class, int.class);
//创建对象并设置属性
Person p1 = (Person) c.newInstance("张三", 20);
System.out.println(p);
System.out.println(p1);
}
}
通过invoke()调用具体方法
method.invoke(obj,args);
obj:具体的实例对象
args:参数的类型,可以是数组
如果是私有方法,一定要打开私有方法的访问权限:
//重点:此处打开私有方法的访问权限,否则会报错
dance.setAccessible(true);
@org.junit.Test
public void test4() throws Exception {
//获取 Person 类的 Class 对象
Class clazz = Class.forName("com.xzy.reflect.Person");
//使用.newInstance 方法创建对象
Person p = (Person) clazz.newInstance();
//如果有参数可以在名字后面指定哪些参数的类型
Method eat = clazz.getDeclaredMethod("eat");
Method dance = clazz.getDeclaredMethod("dance");
//重点:此处打开私有方法的访问权限,否则会报错
dance.setAccessible(true);
eat.invoke(p);
dance.invoke(p);
}
}
运行结果:
this is public method!
this is private method!
Process finished with exit code 0
以上仅为反射简单使用,具体请参考官方文档。