java反射——运行时分析具体对象

概述

在java反射初学篇中,我们了解了可以使用反射获取一个指定的类的信息,只要知道类的名字或者给定一个具体的对象,我们就可以获取到这个类的变量(Field)、方法(Method)、构造方法(Constractor),并且我们可以调用这些内容。

本节我们将学习如何利用反射,分析一个具体的实例化的对象,其目标就是获取其对象的状态。

核心内容

通过类的Class对象,获取指定的Field变量对象,再通过Field对象f,调用get(obj)方法获取指定对象的f所指向的字段,下面用一段代码表述:

Employee harry = new Employee("Harry Hacker",5000);
Class cl = harry.getClass();
Field f = cl.getDeclaredField("name");
Object value = f.get(harry);

上述代码中构建了一个Employee对象,获取它关联"name"的Field对象后,就可以访问其name在运行时的当前状态了,同时你也可以使用f.set(obj,value);来改变值。

访问权

事实上,这段代码有一个小问题,就是加入name是Employee的私有字段,则不能直接就这样用get、set去访问,但却可以用f.setAccessible(true);去获取访问权。

在java 9至java 16中,运行这个程序会出现一些警告信息(WARNING),不影响运行;

而在java 17中则会出现错误(ERROR),解决方法是将java.base模块中的java.util和java.lang包“打开”到“无名模块”。

将来可能会使用可变句柄(variable handle)而不是反射来读写字段。

扩展阅读

以下代码创建了一个ArrayList对象,并使用递归的方法访问它完整的状态。

import java.util.*;

/**
 * This program uses reflection to spy on objects.
 * @version 1.13 2018-03-16
 * @author Cay Horstmann
 */
public class ObjectAnalyzerTest
{
   public static void main(String[] args)
         throws ReflectiveOperationException
   {
      System.out.println(Integer.class.isPrimitive());
      ArrayList<Integer> squares = new ArrayList<Integer>();
      for (int i = 1; i <= 5; i++)
         squares.add(i * i);
      System.out.println(new ObjectAnalyzer().toString(squares));
   }
}
import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.ArrayList;

public class ObjectAnalyzer
{
   private ArrayList<Object> visited = new ArrayList<>();

   /**
    * Converts an object to a string representation that lists all fields.
    * @param obj an object
    * @return a string with the object's class name and all field names and values
    */
   public String toString(Object obj)
         throws ReflectiveOperationException
   {
      if (obj == null) return "null";
      if (visited.contains(obj)) return "...";
      visited.add(obj);
      Class cl = obj.getClass();
      if (cl == String.class) return (String) obj;
      if (cl.isArray())
      {
         String r = cl.getComponentType() + "[]{";
         for (int i = 0; i < Array.getLength(obj); i++)
         {
            if (i > 0) r += ",";
            Object val = Array.get(obj, i);
            if (cl.getComponentType().isPrimitive()) r += val;
            else r += toString(val);
         }
         return r + "}";
      }

      String r = cl.getName() + "\n";
      // inspect the fields of this class and all superclasses
      do
      {
         r += "[";
         Field[] fields = cl.getDeclaredFields();
         AccessibleObject.setAccessible(fields, true);
         // get the names and values of all fields
         for (Field f : fields)
         {
            if (!Modifier.isStatic(f.getModifiers()))
            {
               if (!r.endsWith("[")) r += ",";
               r += f.getName() + "=";
               Class t = f.getType();
               Object val = f.get(obj);
               if (t.isPrimitive()) r += val;
               else r += toString(val);
            }
         }
         r += "]\n";
         cl = cl.getSuperclass();
      }
      while (cl != null);

      return r;
   }
}
java.util.ArrayList
[elementData=class java.lang.Object[]{java.lang.Integer
[value=1]
[]
[]
,java.lang.Integer
[value=4]
[]
[]
,java.lang.Integer
[value=9]
[]
[]
,java.lang.Integer
[value=16]
[]
[]
,java.lang.Integer
[value=25]
[]
[]
,null,null,null,null,null},size=5]
[modCount=5]
[]
[]


进程已结束,退出代码0

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值