getDeclaredFields()-Reflection反射-获取包括父类在内的所有字段

前言

今天Android移动端要加个新功能,所以回归Android程序员的身份.开发的过程中,发现了之前的代码写的有很多问题,真的应该把时间抽出来重构一下了.

其中有反射的一个坑,工具类某方法反射获取传入Model的属性值.但是当我把公共属性抽出来做基类的时候,发现获取不到基类的属性值了.原因是使用了getDeclaredFields();

分析

方法功能
getFields()获取所有public字段,包括父类字段
getDeclaredFields()获取所有字段,public和protected和private,但是不包括父类字段

写个小方法验证一下下~

写两个类,里面定义三个字段,分别用public,protected,private修饰, 
一个叫ParentModel,作为父类. 
一个叫model,继承ParentModel

/**
 * 用作父类
 */
public class ParentModel {

    private String p_privateField;
    public String p_publicField;
    protected  String p_protectedField;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
/**
 * 子类,继承上面定义的用作父类的ParentModel
 */
public class Model extends ParentModel{

    private String privateField;
    public String publicField;
    protected  String protectedField;

}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

ok,分别使用getFields()和getDeclaredFields()获取model的字段,循环打印出来.

 Field[] fs = Model.class.getFields();
 Field[] fs1 = Model.class.getDeclaredFields();
 for (Field f:fs) {          
     Log.d("getFields","getFields---"+f.getName());
 }

 for (Field f:fs1) {           
     Log.d("getDeclaredFields","getDeclaredFields---"+f.getName());
 }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

见证答案的时候到了~ 
getFields()的打印输出:

这里写图片描述

getDeclaredFields()的打印输出:

这里写图片描述

测试证实了我们上面的结论是对的.

我想获取子类和父类的所有Field

如果想用反射通过Model获取parentModel和Model的所有字段,怎么办?很明显上面的两个方法都是满足不了的.那怎么办?

不用怕,我们递归Model的父类去getDeclaredFields(),代码如下:

List<Field> fieldList = new ArrayList<>() ;
Class tempClass = Model.class;
while (tempClass != null) {//当父类为null的时候说明到达了最上层的父类(Object类).
      fieldList.addAll(Arrays.asList(tempClass .getDeclaredFields()));
      tempClass = tempClass.getSuperclass(); //得到父类,然后赋给自己
}
for (Field f : fieldList) {

    Log.d("getAllFields","getFields---"+f.getName());
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

这里写图片描述

可以看到我们获取了Model和ParentModel的全部字段,不仅如此,还多出来了两个字段shadow$_klass_ 和shadow_monitor_,这个是Object中的字段.

shadow$_monitor_shadow$_klass_是Android sdk21之后Object增加的两个字段。

如果你想屏蔽Object类的影响,可以为while循环再添加一个条件:

while (tmpClass !=null && !tmpClass.getName().toLowerCase().equals("java.lang.object") )
{
      ....
}
  • 1
  • 2
  • 3
  • 4

更新说明

2017.6.27更新:

之前被网友 lucky_god88 指出博客反射获取的值和真实情况不符,核实之后,已经更正为正确答案,这里谢谢可爱的lucky_god88 发现并给我指出问题,解决问题的同时自己也在成长。同时也反省自己,以后要代码多加验证,谨慎细致,认真负责。

问题: 
1.getFields() 获取到 protected 类型字段的值

这个原因至今没有再次重现,很奇怪,很费解

2.getFields() 和 getDeclaredFields() 方法反射获取多了一个字段$change

这个和开发工具的配置有关系,好像是因为开启了Instant run 造成的,而且Android Studio 2.2.3已经修复了,链接在这里

参考资料

Retrieving the inherited attribute names/values using Java Reflection

  • 11
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java 反射可以通过以下方式获取父类字段: ```java Class<?> superClass = targetClass.getSuperclass(); // 获取目标类的父类 Field[] fields = superClass.getDeclaredFields(); // 获取父类的所有字段 ``` 其中,`targetClass` 是目标类的 `Class` 对象,`superClass` 是目标类的父类的 `Class` 对象,`fields` 是父类的所有字段数组。使用 `getDeclaredFields()` 可以获取所有的字段包括私有字段。如果只需要获取公有字段,则可以使用 `getFields()` 方法。如果需要获取指定名称的字段,则可以使用 `getDeclaredField(String name)` 方法或 `getField(String name)` 方法。 需要注意的是,如果父类字段是私有的,则需要通过 `setAccessible(true)` 方法来设置其可见性,否则在获取其值时会抛出 `IllegalAccessException` 异常。完整代码如下: ```java public static void main(String[] args) { ChildClass child = new ChildClass(); Class<?> targetClass = child.getClass(); Class<?> superClass = targetClass.getSuperclass(); // 获取目标类的父类 Field[] fields = superClass.getDeclaredFields(); // 获取父类的所有字段 for (Field field : fields) { field.setAccessible(true); // 设置私有字段可见 try { System.out.println(field.getName() + " = " + field.get(child)); // 获取并输出字段的值 } catch (IllegalAccessException e) { e.printStackTrace(); } } } public static class ParentClass { private String parentField = "parentValue"; } public static class ChildClass extends ParentClass { private String childField = "childValue"; } ``` 输出结果为: ``` parentField = parentValue ```
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值