学习Java应该如何理解反射?

标签: Java, 知乎问答

保留所有版权,请引用而不是转载本文(原文地址 https://yeecode.top/blog/31/ )。

首先,对于初学者而言,产生这种疑问是非常正常的。

首先,我们看反射的定义。

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

初学者往往会有这样的的想法:

因此,反射似乎是无用的。

下面我就直接举一个例子,来说明反射的作用。

需求:比较同一个类的两个对象的属性的不同。

方法名为diff,输入参数为两个同类的对象。输出为一个List,其中存放了两个对象的不同的属性的属性名称。

假设User对象,包含name\age\phone三个参数。

List<String> diff(User user1, User user2) {
List<String> result = new ArrayList<>(); 
if(user1.getName() == null && user2.getName() != null || !user1.getName().equals(user2.getName())) {
    result.add("name");
 }
if(user1.getAge() == null && user2.getAge() != null || !user1.getAge().equals(user2.getAge())) {
    result.add("age");
 }
if(user1.getPhone() == null && user2.getPhone() != null || !user1.getPhone().equals(user2.getPhone())) {
    result.add("phone");
 }
}

我们发现一个问题,每个属性都需要分别处理,相同的代码需要些三遍。

当属性更多的话,则要更多遍。

更严重的是,我们的diff方法只能处理User类的对象,不够通用。如果是另外一个School对象,因为属性不同,则要重新编写diff方法。

这时候,就需要反射。

如果使用反射,整个方法可以这样写。

List<String> diff(Object obj1, Object obj2) {
	try{
		Class clazz1 = obj1.getClass();
	    Class clazz2 = obj2.getClass();
	    if(clazz1.equals(clazz2)) {
	    	List<String> result = new ArrayList<>(); 
	    	for (Field field ; clazz1.getDeclaredFields()  ) {
	    		field.setAccessible(true);
	    		Object value1= field.get(obj1);
	    		Object value2= field.get(obj2);
	    		if (value1 == null && value2 != null || !value1.equals(value2)) {
	    			result.add(field.getName());
	    		}
	    	}
	    	return result;
	    } else {
	    	return null;
	    }
	} catch (Exception ex) {
		ex.printStackTrace();
		return null;
	}
}

该方法的优点是可以处理任何类的两个对象,而不需要关心它的属性,十分通用。

例如上述方法可以用在日志记录。

日志记录时,我们可以传入修改前的对象和修改后的对象,通过类似的方法可以直接对比出对象的变化。

例如下面的开源的用户操作日志记录系统就采用了类似的方法实现。大家可以关注该项目学习使用细节。

https://github.com/yeecode/ObjectLogger

反射除了解决对象多样的问题,还能解决外部对象的问题。例如我们引入了一个外部包,可以用反射遍历外部包中对象的属性。

因此,反射是十分重要,有意义的。

可以访问个人知乎阅读更多文章:易哥(https://www.zhihu.com/people/yeecode),欢迎关注。

作者书籍推荐