类的反射
1. 概述
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法。所以先要获取到每一个字节码文件对应的Class类型的对象。
2. 实例
package com.wangjun.jvm;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
/**
* 反射操作练习
*通过反射可以获取:
*1. 类对象
*2. 类的构造器
*3. 类的实例
*4. 类的Filed
*5. 类的方法
*
*通过反射还可以越过泛型检查。泛型用在编译期,编译过后泛型擦除(消失掉)。
*/
public class ReflectClass {
public static void main(String[] args) throws Exception{
//获取class对象的三种方式-------------------------------------
Class classStudent1 = Student.class;//第一种
Student s1 = new Student("wangjun", 25);
Class classStudent2 = s1.getClass();//第二种
//常用这种
Class classStudent3 = Class.forName("com.wangjun.jvm.Student");//第三种
//判断三种方式得到的Class是否是同一个
System.out.println(classStudent1 == classStudent2);
System.out.println(classStudent3 == classStudent2);
//通过反射获取构造函数并使用-------------------------------------
//1. 获取所有“公共”构造方法
Constructor[] constructors = classStudent3.getConstructors();
System.out.println("所有公有的构造方法:");
for(Constructor con: constructors) {
System.out.println(con);
}
//2. 获取所有构造方法(包括公有,私有)
Constructor[] constructors2 = classStudent3.getDeclaredConstructors();
System.out.println("所有构造方法:");
for(Constructor con: constructors2) {
System.out.println(con);
}
//3. 获取公有、无参的构造方法
Constructor constructor1 = classStudent3.getConstructor(null);
//调用构造方法
Object obj1 = constructor1.newInstance();
//4. 获取私有,有参的构造方法
Constructor constructor2 = classStudent3.getDeclaredConstructor(int.class);
constructor2.setAccessible(true);//暴力访问,忽略掉修饰符,不加这一句,后面访问s2.getAge()会报权限错误
Object obj2 = constructor2.newInstance(10);
Student s2 = (Student)obj2;
System.out.println(s2.getAge());
//通过反射获取Field-------------------------------------
Field[] fields1 = classStudent3.getFields();//获取所有公有字段
Field[] fields2 = classStudent3.getDeclaredFields();//获取所有字段
Field field1 = classStudent3.getField("address");//获取公有字段
System.out.println(field1);
Field field2 = classStudent3.getDeclaredField("name");//获取任意字段
field2.setAccessible(true); //暴力访问,解除私有限定,否则field2.set(obj3, "hehe")会报错
Object obj3 = classStudent3.newInstance();
field2.set(obj3, "hehe");
Student s3 = (Student)obj3;
System.out.println(s3.getName());
//通过反射获取方法-------------------------------------
Method[] methods1 = classStudent3.getMethods();
Method[] methods2 = classStudent3.getDeclaredMethods();
Method method1 = classStudent3.getMethod("eat", null);
Method method2 = classStudent3.getDeclaredMethod("run", String.class);
Method method3 = classStudent3.getMethod("eatStatic", null);
method3.invoke(null, null);//调用静态方法时,Object传入null
Object obj4 = classStudent3.newInstance();
method2.setAccessible(true);
method2.invoke(obj4, "xiaoming");//调用方法
//通过反射越过泛型检查-------------------------------------
List<String> list = new ArrayList<>();
list.add("aaa");
list.add("bbb");
//list.add(100); 编译报错
Class listClass = list.getClass();
Method addMethod = listClass.getMethod("add", Object.class);
addMethod.invoke(list, 100);
for(Object o: list) {
System.out.println(o);
}
}
}
辅助类Student
package com.wangjun.jvm;
public class Student {
public String address;
private String name;
private int age;
public Student() {
this.name = "name";
this.age = 100;
}
public Student(String name, int age) {
this.name = name;
this.age = age;
}
private Student(int age) {
this.age = age;
}
public void eat() {
System.out.println("学生在食堂吃饭。");
}
private void run(String str) {
System.out.println(str + "学生在跑步。");
}
public static void eatStatic() {
System.out.println("静态吃饭的方法");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
参考:https://blog.csdn.net/sinat_38259539/article/details/71799078
Last updated