Java运行时使用反射分析对象

本文详细探讨了Java反射机制的工作原理,特别是如何通过调用setAccessible方法绕过访问控制,实现对私有成员的访问。文章还介绍了反射API中的一些关键方法,如getField、getDeclaredField等,并提供了一个使用反射生成通用toString方法的示例。
摘要由CSDN通过智能技术生成

反射机制的默认行为受限于Java的访问控制。然而,如果一个Java程序没有受到安全管理器的控制,就可以覆盖访问控制。为了达到这个目的,需要调用 Field、Method 或constructor 对象的 setAccessible 方法。例如,

f.setAtcessible(true); // now OK to call f.get(harry);

setAccessible方法是 AccessibleObject类中的一个方法, 它是 Field、 Method 和Constructor类的公共超类。这个特性是为调试、持久存储和相似机制提供的。

 

下面是一些常用的方法:

void setAccessible(boolean flag)

为反射对象设置可访问标志。flag 为 true 表明屏蔽 Java 语言的访问检查,使得对象的

私有属性也可以被査询和设置。

boolean isAccessible( )返回反射对象的可访问标志的值。
static void setAccessible(AccessibleObject[] array,boolean flag)是一种设置对象数组可访问标志的快捷方法。

Field getField(String name )

Field[] getField()

返回指定名称的公有域, 或包含所有域的数组

Field getDeclaredField(String name )

Field[] getDeclaredFields( )

返回类中声明的给定名称的域, 或者包含声明的全部域的数组。
Object get(Object obj)返回 obj 对象中用 Field 对象表示的域值。
void set(Object obj ,Object newValue )用一个新值设置 Obj 对象中 Field 对象表示的域。

下面我们用它来编写一个通用的toString方法。

ObjectAnalyzerTest

import java.util.ArrayList;

public class ObjectAnalyzerTest{
	public static void main(String[] args) {
	ArrayList<Integer> squares = new ArrayList<>();
	for (int i = 1; i <= 5; i++)
		squares.add(i * i);
	System.out.println(new ObjectAnalyzer().toString(squares));
   }
}

ObjectAnalyzer

package demo01;


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<>();
	
	public String toString(Object obj) {
	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();
	do {
		r += "[";
		Field[] fields = cl.getDeclaredFields();
		AccessibleObject.setAccessible(fields, true);
		for (Field f : fields) {
			if (!Modifier.isStatic(f.getModifiers())) {
				if (!r.endsWith("[")) r += ",";
				r += f.getName() + "=";
				try {
					Class t = f.getType();
					Object val = f.get(obj);
					if (t.isPrimitive()) r += val;
					else r += toString(val);
				}catch (Exception e) {
					e.printStackTrace();
				}
			}
		}
		r += "]";
		cl = cl.getSuperclass();
	}
	while (cl != null);
	return r;
	}
}

将输出以下结果:

ArrayList<Integer> squares = new ArayList<>();
for (int i = 1; i <= 5; i++) squares.add(i*i);
System.out .println(new ObjectAnal yzer().toString(squares));
将会产生下时的打印结果:
java.uti1.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=2 5][][].null.null,null,null.null},size=5][modCount=5][] []

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值