Java反射原理:将普通java类的各个组成部分分别封装为对象,包括类的成员变量、成员方法、成员构造。
其所对应的对象分别是:Field、Method、Constructor。
通过类的字节码对象,即Class
对象获取
Class
类将所有的普通java类作为对象。
三个阶段获取Class对象的方法
Class.forName(“全类名”)
将字节码文件加载进内存,并返回class对象。多用于配置文件
类名.class
通过类名的class属性获取,该方法常用于参数的传递。
对象名.getClass()
getClass()方法定义于Object类中,用于对象获取字节码。
同一个字节码文件(*.class)在程序运行过程中,只会被加载一次。无论通过哪种方法获取的class对象都是同一个。
获取成员变量
方法名 | 作用 |
---|---|
Field[] getFields() | 获取所有public修饰的成员变量 |
Field getField(String name) | 获取指定名称的 public修饰的成员变量 |
Field[] getDeclaredFields() | 获取所有的成员变量,不考虑修饰符 |
Field getDeclaredField(String name) | 获取指定名称的成员变量,不考虑修饰符 |
方法名 | 作用 |
---|---|
void set(Object obj, Object value) | 设置值,将指定对象变量上此 Field 对象表示的字段设置为指定的新值。 |
Object get(Object obj) | 获取值,返回指定对象上此 Field 表示的字段的值 |
setAccessible(true) | 忽略访问权限修饰符的安全检查 |
当访问私有变量时,必须设置setAccessible(true)来忽略访问权限修饰符的安全检查,否则没有权限访问。
案例
public static void main(String[] args) throws Exception{
Student student=new Student(); //新建一个对象
student.setId("01"); //设置改对象的id值
student.setName("tom"); //设置改对象的name值
student.setBirthday("2020-03-05"); //设置改对象的Birthday值
student.setSex("男"); //设置改对象的性别值
Class> clazz =Class.forName("Student"); //获取该类
Class> clazz2=Student.class; //类.class
Class> clazz3=student.getClass(); //对象.getclass
Field field = null;
field = clazz.getDeclaredField("id");//获取类的指定属性包含私有属性
int modifiers =field.getModifiers(); //获取类的指定属性的修饰符
if(Modifier.isPrivate(modifiers)){ //指定属性的修饰符是否为private
field.setAccessible(true); //如果属性为private就设置为可以访问
}
Object id =field.get(student); //获取该对象的属性值
if(id instanceof String){
System.out.println("id = " + id);
field.set(student,"002"); //更改该对象的属性值
System.out.println("id = " + student.getId());
}
}
student类
public class Student {
private String id;
private String name;
private String sex;
private String birthday;
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public String getBirthday() {
return birthday;
}
public void setBirthday(String birthday) {
this.birthday = birthday;
}
}
获取构造方法
方法名 | 作用 |
---|---|
Constructor>[] getConstructors() | 返回此 Class 对象所表示的类的所有公共构造方法。 |
Constructor |
返回此 Class 对象所表示的类的指定参数公共构造方法。 |
Constructor>[] getDeclaredConstructors() | 返回此 Class 对象所表示的类的所有构造方法,不考虑修饰符。 |
Constructor |
返回此 Class 对象所表示的类的指定参数构造方法,不考虑修饰符。 |
Constructor类中的常用方法
方法名 | 作用 |
---|---|
T newInstance(Object... initargs) | 实例化一个对象 |
Class cls = Student.class;
//方法一
Constructor c1 = cls.getConstructor(String.class,String.class,String.class,String.class);
Object s1 = c1.newInstance("帝释天","24","看小说","001");
System.out.println(s1);
//方法二 调用空参构造
Constructor c2 = cls.getConstructor();
Object s2 = c2.newInstance();
System.out.println(s2);
//方法三 由Class对象直接创建
Object c3 = cls.newInstance();
System.out.println(c3);
获取成员方法
方法名 | 作用 |
---|---|
Method[] getMethods() | 获取所有public修饰的成员方法 |
Method getMethod(String name, 类>... parameterTypes) | 获取指定参数形式的 public修饰的成员方法 |
Method[] getDeclaredMethods() | 获取所有成员方法,不考虑修饰符 |
Method getDeclaredMethod(String name, 类>... parameterTypes) | 获取指定参数形式的成员方法,不考虑修饰符 |
Method类中的常用方法
方法名 | 作用 |
---|---|
Object invoke(Object obj, Object... args) | 执行方法,对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。 |
String getName | 获取方法名 |
Class cls = Student.class;
Student stu = new Student();
//执行带参数方法
Method song1 = cls.getMethod("song", String.class);
song1.invoke(stu, "你确定这就是爱吗");
//执行无参数方法
Method song2 = cls.getMethod("song");
song2.invoke(stu);
//获取方法名
Method[] methods = cls.getMethods();
for (Method method : methods) {
System.out.println(method);
}
public void song() {
System.out.println("We are the word...");
}
public void song(String songs) {
System.out.println("我会唱"+songs);
}
invoke(Object obj, Object... args) 对该对象,执行该方法;
package com.etoppaas.insureagent.common.utils;
import com.etoppaas.common.tools.EmptyUtils;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* 比较相同类的相同属性值是否相同
*/
public class CompareClassUtil {
public static Boolean compareTwoClass(Object class1, Object class2, List ignoreFieldList) throws ClassNotFoundException, IllegalAccessException {
//动态的获取指定对象的class
Class> clazz1 = class1.getClass();
Class> clazz2 = class2.getClass();
// 获取类中所有的属性(public、protected、default、private),但不包括继承的属性,返回 Field 对象的一个数组
Field[] field1 = clazz1.getDeclaredFields();
Field[] field2 = clazz2.getDeclaredFields();
//遍历属性列表field1
for (int i = 0; i < field1.length; i++) {
//遍历属性列表field2
for (int j = 0; j < field2.length; j++) {
//如果field1[i]属性名与field2[j]属性名内容相同
if (field1[i].getName().equals(field2[j].getName())) {
Boolean checkIgnoreField = true;
if (EmptyUtils.isNotEmpty(ignoreFieldList)) {
for (int ignoreFieldNum = 0; ignoreFieldNum < ignoreFieldList.size(); ignoreFieldNum++) {
//如果要忽略的类型里存在,就跳过这个方法
if (field1[i].getName().equals(ignoreFieldList.get(ignoreFieldNum))) {
checkIgnoreField = false;
}
}
if (checkIgnoreField) {
//调过本次循环的下面语句执行
continue;
}
if (!compareTwo(field1[i], field2[j], class1, class2)) {
return false;
}
break;
} else {
if (!compareTwo(field1[i], field2[j], class1, class2)) {
return false;
}
break;
}
}
}
}
return true;
}
/**
* 对比两个数据是否内容相同
*
* @param
* @return boolean类型
*/
public static boolean compareTwo(Field field1, Field field2, Object class1, Object class2) throws IllegalAccessException {
//让我们可以访问私有变量的值
field1.setAccessible(true);
field2.setAccessible(true);
//如果field1[i]属性值与field2[j]属性值内容不相同
//为了不重写equals方法目前只能比较基础的类型
if ("java".equals(field1.getGenericType().toString())) {
}
//返回该类下面对应的该属性值,并返回结果
Object object1 = field1.get(class1);
Object object2 = field2.get(class2);
if (object1 == null && object2 == null) {
return true;
}
if (object1 == null && object2 != null) {
return false;
}
if (object1.equals(object2)) {
return true;
}
return false;
}
}