文档章节

用反射的方式获取父类中的所有属性和方法

bharals
 bharals
发布于 2017/05/19 17:04
字数 1138
阅读 864
收藏 0

码上生花,ECharts 作品展示赛正式启动!>>>

package com.syh.jdbc.reflection_super;

/** * 父类 * @author syh * */

public class Parent {

    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.syh.jdbc.reflection_super;

/** * 子类 * @author syh * */

public class Son extends Parent{

}
package com.syh.jdbc.reflection_super;

import java.lang.reflect.Field;

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

/** * 方法类 * @author syh * */

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.syh.jdbc.reflection_super;

import static org.junit.Assert.*;

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

import org.junit.Test;

/** * 测试类,用JUnit4 进行测试 * @author syh * */

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"));
    }

}

© 著作权归作者所有

bharals
粉丝 0
博文 26
码字总数 46307
作品 0
私信 提问
加载中
请先登录后再评论。
反射的作用:通过反射获取属性和方法

今天主要来谈一谈反射中获取成员方法和成员变量的主要方法: 在获取之前我们首先创造两个类 Person类(父类): 1 public class Person { 2 private String name;//私有属性 3 public Integ...

osc_oupyj5vz
04/16
2
0
框架原理那点事--不就反射嘛。

记得之前写了一篇json转实体的文章 <!-- more --> 记得之前写了一篇json转实体的文章,里面就用到了java反射的文章,但是当时只是了解到反射这个知识点,并没有深入的了解,寒假闲来无事觉顶...

osc_kwrafgul
04/16
6
0
Java核心技术梳理-类加载机制与反射

一、引言 反射机制是一个非常好用的机制,C#和Java中都有反射,反射机制简单来说就是在程序运行状态时,对于任意一个类,能够知道这个类的所有属性和方法,对于任意一个对象,能够调用它的任...

osc_6oe4sgbd
04/16
3
0
Java反射机制(未完成,还缺最后一个)

1、背景 1)Reflection也就是反射 是Java被视为动态(或准动态)语言的一个关键性质 2)反射机制指的是程序在运行时能够获取任何类的内部所有信息 2、实现功能概述 1)只要给定类的全名,即可...

萧小蚁
2015/05/03
259
0
java反射(三)--反射与操作类

一.反射与操作类   在反射机制的处理过程之中不仅仅只是一个实例化对象的处理操作,更多的情况下还有类的组成的操作,任何一个类的基本组成结构:父类(父接口),包,属性,方法(构造方法,普通方法...

osc_dwomrlil
2019/08/28
4
0

没有更多内容

加载失败,请刷新页面

加载更多

HashMap解析(主要JDK1.8,附带1.7出现的问题以及区别)

按问题的形式来吧,这些大多是我自己总结的,如有错误请及时指正谢谢 1.你了解HashMap么,可以说说么?   首先,HashMap是一种数据结构,可以快速的帮我们存取数据。它的底层数据结构在1.7...

osc_gzy7qjj6
47分钟前
16
0
CentOS8.1操作系下使用通用二进制包安装MySQL8.0(实践整理自MySQL官方)

  写在前的的话: 在IT技术日新月异的今天,老司机也可能在看似熟悉的道路上翻车,甚至是大型翻车现场!自己一个人开车过去翻个车不可怕,可怕的是带着整个团队甚至是整个公司一起翻车山崖...

osc_494omtst
48分钟前
15
0
关于对健壮性代码的理解

这两天学到了很多知识,对项目的严密性有了极为深刻的理解,简而言之,身为前端开发者要站在用户的角度去写相关代码,而不能仅仅局限于理所当然,也不可以认为数据有便有,没有便没有,身为开...

osc_47pscir3
50分钟前
13
0
【Spring注解驱动开发】组件注册-@ComponentScan-自动扫描组件&指定扫描规则

写在前面 在实际项目中,我们更多的是使用Spring的包扫描功能对项目中的包进行扫描,凡是在指定的包或子包中的类上标注了@Repository、@Service、@Controller、@Component注解的类都会被扫描...

osc_a5w6ccj0
51分钟前
10
0
学习c++ (五) 用duilib 编写界面

还是那句话,不想用MFC,写的界面丑,不想依赖MFC的一些库,在网上闲逛,发现duilib还可以,至于QT,有时间再研究, dulib已经没人更新了,估计也是翻不出什么花样了,但研究一下总算是C++的...

osc_jklrr90y
52分钟前
19
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部