java比较两个类的值不相同_java 反射---------比较两个相同类型的对象相同属性的属性值是否相同的具体调用...

本文介绍了Java反射的概念,包括反射结构、相关函数的使用,如通过Class对象获取成员变量、构造函数和方法。接着,通过示例展示了如何获取并设置私有属性,以及如何实例化对象和调用方法。最后,提供了一个比较工具类`CompareClassUtil`,用于比较两个相同类的相同属性值是否相同,考虑了忽略特定字段的情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一.Java反射的概念:

在Java运行时环境中,对于任意一个类,可以知道这个类有哪些属性和方法。 对于任意一个对象,可以调用它的任意一个方法。 这种动态获取类的信息以及动态调用对象的方法的功能来自于Java 语言的反射(Reflection)机制。

反射结构分析

Java反射原理:将普通java类的各个组成部分分别封装为对象,包括类的成员变量、成员方法、成员构造。

其所对应的对象分别是:Field、Method、Constructor。

2.反射的相关函数,用法

一.class对象

通过类的字节码对象,即Class对象获取

Class类将所有的普通java类作为对象。

180db003adb723fe51efffc0db72b007.png

三个阶段获取Class对象的方法

Class.forName(“全类名”)

将字节码文件加载进内存,并返回class对象。多用于配置文件

类名.class

通过类名的class属性获取,该方法常用于参数的传递。

对象名.getClass()

getClass()方法定义于Object类中,用于对象获取字节码。

同一个字节码文件(*.class)在程序运行过程中,只会被加载一次。无论通过哪种方法获取的class对象都是同一个。

Class对象中的常用方法

获取成员变量

方法名

作用

Field[] getFields()

获取所有public修饰的成员变量

Field getField(String name)

获取指定名称的 public修饰的成员变量

Field[] getDeclaredFields()

获取所有的成员变量,不考虑修饰符

Field getDeclaredField(String name)

获取指定名称的成员变量,不考虑修饰符

Field类的常用方法

方法名

作用

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 对象所表示的类的所有公共构造方法。

ConstructorgetConstructor(类>... parameterTypes)

返回此 Class 对象所表示的类的指定参数公共构造方法。

Constructor>[] getDeclaredConstructors()

返回此 Class 对象所表示的类的所有构造方法,不考虑修饰符。

ConstructorgetDeclaredConstructor(类>... parameterTypes)

返回此 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) 对该对象,执行该方法;

3.比较的相关工具类的书写

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, ListignoreFieldList) 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;

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值