学习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),欢迎关注。