反射工具类(调用父类的方法和字段)

本文详细介绍了一个用于操作Java类、方法及字段的工具类,利用Java反射API实现了无视访问权限限制来调用方法和修改字段的功能,并提供了具体示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用这个工具类,可以完成父类,基类,方法,字段,无论什么权限都可以调用.

package com.reflect;
/**
 * 基类
 * @author jianghui
 */
public class GrandParent {
    public String publicField = "1";

    String defaultField = "2";

    protected String protectedField = "3";

    private String privateField = "4";

    public void publicMethod() {
        System.out.println("publicMethod...");
    }

    void defaultMethod() {
        System.out.println("defaultMethod...");
    }

    protected void protectedMethod() {
        System.out.println("protectedMethod...");
    }

    private void privateMethod() {
        System.out.println("privateMethod...");
    }

}
package com.reflect;
/**
 * 父类
 * @author jianghui
 */
public class Parent extends GrandParent {

}
package com.reflect;
/**
 * 子类
 * @author jianghui
 */
public class Son extends Parent {

}
package com.reflect;

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * 方法类
 * 
 * @author jianghui
 */
public class ReflectionUtils {

    /**
     * 循环向上转型, 获取对象的 DeclaredMethod
     * 
     * @param object
     *            : 子类对象
     * @param methodName
     *            : 父类中的方法名
     * @param parameterTypes
     *            : 父类中的方法参数类型
     * @return 父类中的方法对象
     */

    public static Method getDeclaredMethod(Object object, String methodName, Class<?>... parameterTypes) {
        Method method = null;

        for (Class<?> clazz = object.getClass(); clazz != Object.class; clazz = clazz.getSuperclass()) {
            try {
                method = clazz.getDeclaredMethod(methodName, parameterTypes);
                return method;
            } catch (Exception e) {
                // 这里甚么都不要做!并且这里的异常必须这样写,不能抛出去。
                // 如果这里的异常打印或者往外抛,则就不会执行clazz=clazz.getSuperclass(),最后就不会进入到父类中了
            }
        }

        return null;
    }

    /**
     * 直接调用对象方法, 而忽略修饰符(private, protected, default)
     * 
     * @param object
     *            : 子类对象
     * @param methodName
     *            : 父类中的方法名
     * @param parameterTypes
     *            : 父类中的方法参数类型
     * @param parameters
     *            : 父类中的方法参数
     * @return 父类中方法的执行结果
     */

    public static Object invokeMethod(Object object, String methodName, Class<?>[] parameterTypes,
            Object[] parameters) {
        // 根据 对象、方法名和对应的方法参数 通过反射 调用上面的方法获取 Method 对象
        Method method = getDeclaredMethod(object, methodName, parameterTypes);

        // 抑制Java对方法进行检查,主要是针对私有方法而言
        method.setAccessible(true);

        try {
            if (null != method) {

                // 调用object 的 method 所代表的方法,其方法的参数是 parameters
                return method.invoke(object, parameters);
            }
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        return null;
    }

    /**
     * 循环向上转型, 获取对象的 DeclaredField
     * 
     * @param object
     *            : 子类对象
     * @param fieldName
     *            : 父类中的属性名
     * @return 父类中的属性对象
     */

    public static Field getDeclaredField(Object object, String fieldName) {
        Field field = null;

        Class<?> clazz = object.getClass();

        for (; clazz != Object.class; clazz = clazz.getSuperclass()) {
            try {
                field = clazz.getDeclaredField(fieldName);
                return field;
            } catch (Exception e) {
                // 这里甚么都不要做!并且这里的异常必须这样写,不能抛出去。
                // 如果这里的异常打印或者往外抛,则就不会执行clazz = clazz.getSuperclass(),最后就不会进入到父类中了

            }
        }

        return null;
    }

    /**
     * 直接设置对象属性值, 忽略 private/protected 修饰符, 也不经过 setter
     * 
     * @param object
     *            : 子类对象
     * @param fieldName
     *            : 父类中的属性名
     * @param value
     *            : 将要设置的值
     */

    public static void setFieldValue(Object object, String fieldName, Object value) {

        // 根据 对象和属性名通过反射 调用上面的方法获取 Field对象
        Field field = getDeclaredField(object, fieldName);

        // 抑制Java对其的检查
        field.setAccessible(true);

        try {
            // 将 object 中 field 所代表的值 设置为 value
            field.set(object, value);
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

    }

    /**
     * 直接读取对象的属性值, 忽略 private/protected 修饰符, 也不经过 getter
     * 
     * @param object
     *            : 子类对象
     * @param fieldName
     *            : 父类中的属性名
     * @return : 父类中的属性值
     */

    public static Object getFieldValue(Object object, String fieldName) {

        // 根据 对象和属性名通过反射 调用上面的方法获取 Field对象
        Field field = getDeclaredField(object, fieldName);

        // 抑制Java对其的检查
        field.setAccessible(true);

        try {
            // 获取 object 中 field 所代表的属性值
            return field.get(object);

        } catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }
}
package com.reflect;

import java.lang.reflect.Field;
import java.lang.reflect.Method;

import org.junit.Test;

/**
 * 测试类 
 * @author jianghui
 */
public class ReflectionUtilsTest {  

   /** 
    * 测试获取父类的各个方法对象 
    */  
   @Test  
   public void testGetDeclaredMethod() {  

       Object obj = new Son() ;  

       //获取公共方法名  
       Method publicMethod = ReflectionUtils.getDeclaredMethod(obj, "publicMethod") ;  
       System.out.println(publicMethod.getName());  

       //获取默认方法名  
       Method defaultMethod = ReflectionUtils.getDeclaredMethod(obj, "defaultMethod") ;  
       System.out.println(defaultMethod.getName());  

       //获取被保护方法名  
       Method protectedMethod = ReflectionUtils.getDeclaredMethod(obj, "protectedMethod") ;  
       System.out.println(protectedMethod.getName());  

       //获取私有方法名  
       Method privateMethod = ReflectionUtils.getDeclaredMethod(obj, "privateMethod") ;  
       System.out.println(privateMethod.getName());  
   }  

   /** 
    * 测试调用父类的方法 
    * @throws Exception 
    */  

   @Test  
   public void testInvokeMethod() throws Exception {  
       Object obj = new Son() ;  

       //调用父类的公共方法  
       ReflectionUtils.invokeMethod(obj, "publicMethod", null , null) ;  

       //调用父类的默认方法  
       ReflectionUtils.invokeMethod(obj, "defaultMethod", null , null) ;  

       //调用父类的被保护方法  
       ReflectionUtils.invokeMethod(obj, "protectedMethod", null , null) ;  

       //调用父类的私有方法  
       ReflectionUtils.invokeMethod(obj, "privateMethod", null , null) ;  
   }  

   /** 
    * 测试获取父类的各个属性名 
    */  

   @Test  
   public void testGetDeclaredField() {  

       Object obj = new Son() ;  

       //获取公共属性名  
       Field publicField = ReflectionUtils.getDeclaredField(obj, "publicField") ;  
       System.out.println(publicField.getName());  

       //获取默认属性名  
       Field defaultField = ReflectionUtils.getDeclaredField(obj, "defaultField") ;  
       System.out.println(defaultField.getName());  

       //获取受保护属性名  
       Field protectedField = ReflectionUtils.getDeclaredField(obj, "protectedField") ;  
       System.out.println(protectedField.getName());  

       //获取私有属性名  
       Field privateField = ReflectionUtils.getDeclaredField(obj, "privateField") ;  
       System.out.println(privateField.getName());  

   }  

   @Test  
   public void testSetFieldValue() {  

       Object obj = new Son() ;  

       System.out.println("原来的各个属性的值: ");  
       System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField"));  
       System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField"));  
       System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField"));  
       System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField"));  

       ReflectionUtils.setFieldValue(obj, "publicField", "a") ;  
       ReflectionUtils.setFieldValue(obj, "defaultField", "b") ;  
       ReflectionUtils.setFieldValue(obj, "protectedField", "c") ;  
       ReflectionUtils.setFieldValue(obj, "privateField", "d") ;  

       System.out.println("***********************************************************");  

       System.out.println("将属性值改变后的各个属性值: ");  
       System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField"));  
       System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField"));  
       System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField"));  
       System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField"));  

   }  

  /**
   * 获取字段的值
   */
   @Test  
   public void testGetFieldValue() {  

       Object obj = new Son() ;  

       System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField"));  
       System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField"));  
       System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField"));  
       System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField"));  
   }  

}  
<think>好的,我现在需要解决用户的问题:如何通过Java反射机制获取多级父类中的指定字段,即使当前类中没有该字段。首先,我需要回忆Java反射相关的知识,特别是关于访问父类字段的部分。 Java反射API中,Class类提供了getField()getDeclaredField()方法,但getField()只能获取公共字段,包括继承的,而getDeclaredField()只能获取当前类声明的字段,不包括父类。所以,如果字段不在当前类,但可能在父类中,需要手动遍历父类层级。 用户的需求是获取多级父类中的字段,所以需要循环检查每个父类,直到找到字段或到达Object类为止。具体步骤如下: 1. 从当前类的Class对象开始。 2. 使用getDeclaredField()尝试获取字段,如果找到则返回。 3. 如果当前类没有,获取其父类,并重复步骤2。 4. 直到父类为null(即到达Object的父类)时停止,抛出异常。 需要注意的是,getDeclaredField()可能抛出NoSuchFieldException,所以需要在循环中捕获异常,继续查找父类。此外,需要处理访问权限的问题,可能需要调用setAccessible(true)来访问私有字段。 现在,结合代码示例,编写一个工具方法,递归或循环地查找字段。例如: public static Field getFieldInHierarchy(Class<?> clazz, String fieldName) throws NoSuchFieldException { while (clazz != null) { try { return clazz.getDeclaredField(fieldName); } catch (NoSuchFieldException e) { clazz = clazz.getSuperclass(); } } throw new NoSuchFieldException(fieldName); } 这样,如果在当前类找不到,就继续在父类中查找。需要确保正确处理异常,直到父类为null为止。 另外,需要考虑安全权限的问题,如果安全管理器不允许访问私有字段,可能需要设置访问权限。但在大多数情况下,调用setAccessible(true)可以绕过访问检查。 总结步骤:遍历类层级,使用getDeclaredField尝试获取,找不到则向上查找父类,直到找到或抛出异常。这样就能实现获取多级父类中的字段。</think>要通过Java反射机制获取多级父类中的指定字段,需遍历类继承层级并递归查找父类字段。以下是分步实现方法: --- ### 步骤 1:理解反射获取字段的限制 - `Class.getField()` 仅能获取当前类或父类的 **public字段** - `Class.getDeclaredField()` 可获取当前类的 **所有字段(包括private)**,但无法直接访问父类字段[^2][^3] --- ### 步骤 2:实现多级父类字段查找 ```java public static Field getFieldInHierarchy(Class<?> targetClass, String fieldName) throws NoSuchFieldException { Class<?> currentClass = targetClass; while (currentClass != null) { // 终止条件:到达Object父类 try { // 尝试获取当前类的字段(含非public字段) Field field = currentClass.getDeclaredField(fieldName); field.setAccessible(true); // 突破private访问限制 return field; } catch (NoSuchFieldException e) { // 未找到则向上查找父类 currentClass = currentClass.getSuperclass(); } } throw new NoSuchFieldException( "字段 '" + fieldName + "' 在类 " + targetClass.getName() + " 及其父类中未找到" ); } ``` --- ### 步骤 3:使用示例 ```java class Grandparent { private String familySecret = "Confidential"; } class Parent extends Grandparent {} class Child extends Parent {} public static void main(String[] args) { try { Field secretField = getFieldInHierarchy(Child.class, "familySecret"); System.out.println("字段类型: " + secretField.getType().getSimpleName()); // 输出: String } catch (NoSuchFieldException e) { e.printStackTrace(); } } ``` --- ### 关键点解析 1. **继承层级遍历**:通过 `getSuperclass()` 方法逐级向上查找父类 2. **访问控制突破**:`setAccessible(true)` 可访问私有字段(需安全管理器允许) 3. **异常处理**:未找到字段时抛出明确的异常信息 --- ### 注意事项 - 若字段在多个父类中存在同名冲突,此方法返回 **最先找到的字段** - 查找静态字段时需确保类已完成初始化[^3] - 反射性能开销较高,建议缓存Field对象 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值