JAVA 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 java 语言的反射机制。
实际上,我们创建的每一个类也都是对象,即类本身是 java.lang.Class 类的实例对象。这个实例对象称之为类对象,也就是 Class 对象。
反射就是把 java 类中的各种成分映射成一个个的 Java 对象
例如:一个类有:成员变量、方法、构造方法、包等等信息,利用反射技术可以对一个类进行解剖,把个个组成部分映射成一个个对象。
(其实:一个类中这些成员方法、构造方法、在加入类中都有一个类来描述)
如图是类的正常加载过程:反射的原理在与 class 对象。
熟悉一下加载的时候:Class 对象的由来是将 class 文件读入内存,并为之创建一个 Class 对象。
package test.reflect;
/**
* @author 谢世杰
*/
public class GetClass {
public static void main(String[] args) {
Student student = new Student();
// 第一种方式
Class stuClass1 = student.getClass();
System.out.println(stuClass1.getName());
// 第二种方式
Class stuClass2 = Student.class;
System.out.println(int.class.getName());
System.out.println(stuClass1 == stuClass2);
// 第三种方式
try {
Class stuClass3 = Class.forName("test.reflect.Student");
System.out.println(stuClass1 == stuClass3);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
运行结果:
test.reflect.Student
int
true
true
可以看到所有获得到的class对象都是同一个。
package test.reflect;
/**
* @author 谢世杰
*/
public class Student {
public String id;
private String name;
private String sex;
private int age;
public Student() {
System.out.println("公有,无参构造函数");
}
public Student(String name) {
System.out.println("name:" + name);
}
private Student(int age) {
System.out.println("私有构造函数" + age);
}
public void test(String name, int age) {
System.out.println("name:"+name+" age:"+age);
}
public void setName(String name) {
this.name = name;
}
public void setSex(String sex) {
this.sex = sex;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public String getSex() {
return sex;
}
public int getAge() {
return age;
}
}
package test.reflect;
import java.lang.reflect.Constructor;
/**
* @author 谢世杰
*/
public class Constructors {
public static void main(String[] args) throws Exception {
Class stuClass = Class.forName("test.reflect.Student");
System.out.println("===获得所有公有构造方法===");
Constructor[] cons = stuClass.getConstructors();
for (Constructor con : cons) {
System.out.println(con);
}
System.out.println("===获得所有构造方法===");
Constructor[] cons2 = stuClass.getDeclaredConstructors();
for (Constructor con : cons2) {
System.out.println(con);
}
System.out.println("===根据参数类型获得指定构造函数===");
Constructor conNull = stuClass.getDeclaredConstructor(null);
System.out.println("无参构造函数" + conNull);
Constructor conInt = stuClass.getDeclaredConstructor(int.class);
System.out.println("参数为int的构造函数" + conInt);
Constructor conString = stuClass.getDeclaredConstructor(String.class);
System.out.println("参数为String构造函数" + conString);
System.out.println("===调用构造函数===");
Object obj = conNull.newInstance();
// 注意,私有构造函数需要设置使用权限,否则报错
conInt.setAccessible(true);
Object obj2 = conInt.newInstance(10);
}
}
运行结果:
===获得所有公有构造方法===
public test.reflect.Student(java.lang.String)
public test.reflect.Student()
===获得所有构造方法===
private test.reflect.Student(int)
public test.reflect.Student(java.lang.String)
public test.reflect.Student()
===根据参数类型获得指定构造函数===
无参构造函数public test.reflect.Student()
参数为int的构造函数private test.reflect.Student(int)
参数为String构造函数public test.reflect.Student(java.lang.String)
===调用构造函数===
公有,无参构造函数
私有构造函数10
package test.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
/**
* @author 谢世杰
*/
public class Fields {
public static void main(String[] args) throws Exception {
Class stuClass = Class.forName("test.reflect.Student");
System.out.println("===所有公有字段===");
Field[] fields1 = stuClass.getFields();
for (Field field : fields1) {
System.out.println(field);
}
System.out.println("===所有字段===");
Field[] fields2 = stuClass.getDeclaredFields();
for (Field field : fields2) {
System.out.println(field);
}
System.out.println("===根据字段名获取字段并使用===");
// 获得id字段
Field field = stuClass.getField("id");
// 私有字段需要开放权限,公有不需要处理
field.setAccessible(true);
// 获取构造函数
Constructor constructor = stuClass.getConstructor();
Object obj = constructor.newInstance();
// 设置字段值
field.set(obj,"xsj");
// 查看结果
Student stu = (Student)obj;
System.out.println(stu.id);
}
}
运行结果:
===所有公有字段===
public java.lang.String test.reflect.Student.id
===所有字段===
public java.lang.String test.reflect.Student.id
private java.lang.String test.reflect.Student.name
private java.lang.String test.reflect.Student.sex
private int test.reflect.Student.age
===根据字段名获取字段并使用===
公有,无参构造函数
xsj
设置字段的值:
Field --> public void set(Object obj,Object value):
参数说明:
1.obj:要设置的字段所在的对象;
2.value:要为字段设置的值;
package test.reflect;
import java.lang.reflect.Method;
/**
* @author 谢世杰
*/
public class Methods {
public static void main(String[] args) throws Exception{
Class stuClass = Class.forName("test.reflect.Student");
// 这里获取的方法包括父类的方法
System.out.println("===获取所有公有方法(包括继承的父类的方法)===");
Method[] methods1 = stuClass.getMethods();
for (Method method : methods1) {
System.out.println(method);
}
System.out.println("===获取本类所有方法===");
Method[] methods2 = stuClass.getDeclaredMethods();
for (Method method : methods2) {
System.out.println(method);
}
System.out.println("===获取指定方法并使用===");
Method method = stuClass.getMethod("test",String.class,int.class);
Object obj = stuClass.getConstructor().newInstance();
// 调用函数
method.invoke(obj,"xsj",123);
}
}
运行结果:
===获取所有公有方法(包括继承的父类的方法)===
public java.lang.String test.reflect.Student.getName()
public void test.reflect.Student.setName(java.lang.String)
public void test.reflect.Student.test(java.lang.String,int)
public void test.reflect.Student.setAge(int)
public void test.reflect.Student.setSex(java.lang.String)
public java.lang.String test.reflect.Student.getSex()
public int test.reflect.Student.getAge()
public final void java.lang.Object.wait() throws java.lang.InterruptedException
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
public boolean java.lang.Object.equals(java.lang.Object)
public java.lang.String java.lang.Object.toString()
public native int java.lang.Object.hashCode()
public final native java.lang.Class java.lang.Object.getClass()
public final native void java.lang.Object.notify()
public final native void java.lang.Object.notifyAll()
===获取本类所有方法===
public java.lang.String test.reflect.Student.getName()
public void test.reflect.Student.setName(java.lang.String)
public void test.reflect.Student.test(java.lang.String,int)
public void test.reflect.Student.setAge(int)
public void test.reflect.Student.setSex(java.lang.String)
public java.lang.String test.reflect.Student.getSex()
public int test.reflect.Student.getAge()
===获取指定方法并使用===
公有,无参构造函数
name:xsj age:123
调用方法:
Method --> public Object invoke(Object obj,Object… args):
参数说明:
obj : 要调用方法的对象;
args:调用方式时所传递的实参
student:
public class Student {
public void show(){
System.out.println("is show()");
}
}
配置文件(以TXT为例) :
className = cn.fanshe.Student
methodName = show
测试类
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.lang.reflect.Method;
import java.util.Properties;
/*
* 我们利用反射和配置文件,可以使:应用程序更新时,对源码无需进行任何修改
* 我们只需要将新类发送给客户端,并修改配置文件即可
*/
public class Demo {
public static void main(String[] args) throws Exception {
//通过反射获取Class对象
Class stuClass = Class.forName(getValue("className"));//"cn.fanshe.Student"
//2获取show()方法
Method m = stuClass.getMethod(getValue("methodName"));//show
//3.调用show()方法
m.invoke(stuClass.getConstructor().newInstance());
}
//此方法接收一个key,在配置文件中获取相应的value
public static String getValue(String key) throws IOException{
Properties pro = new Properties();//获取配置文件的对象
FileReader in = new FileReader("pro.txt");//获取输入流
pro.load(in);//将流加载到配置文件对象中
in.close();
return pro.getProperty(key);//返回根据key获取的value值
}
}
控制台输出:
is show()
需求:
当我们升级这个系统时,不要 Student 类,而需要新写一个 Student2 的类时,这时只需要更改 pro.txt 的文件内容就可以了。代码就一点不用改动
student2 类:
public class Student2 {
public void show2(){
System.out.println("is show2()");
}
}
配置文件更改为:
className = cn.fanshe.Student2
methodName = show2
控制台输出:
is show2();
泛型用在编译期,编译过后泛型擦除(消失掉)。所以是可以通过反射越过泛型检查的
import java.lang.reflect.Method;
import java.util.ArrayList;
/*
* 通过反射越过泛型检查
*
* 例如:有一个String泛型的集合,怎样能向这个集合中添加一个Integer类型的值?
*/
public class Demo {
public static void main(String[] args) throws Exception{
ArrayList<String> strList = new ArrayList<>();
strList.add("aaa");
strList.add("bbb");
// strList.add(100);
//获取ArrayList的Class对象,反向的调用add()方法,添加数据
Class listClass = strList.getClass(); //得到 strList 对象的字节码 对象
//获取add()方法
Method m = listClass.getMethod("add", Object.class);
//调用add()方法
m.invoke(strList, 100);
//遍历集合
for(Object obj : strList){
System.out.println(obj);
}
}
}
控制台输出:
aaa
bbb
100