傻瓜教你反射

作者: 吃啥呀 | 来源:发表于2018-07-15 21:47 被阅读43次

    Field

    1) Class -> Field

    Class.getDeclaredField(String s)
    返回一个 Field 对象,该对象反映此Class对象所表示的类或接口的指定已声明字段(包括私有成员
    Class.getDeclaredFields()
    返回 Field 对象的一个数组,该数组包含此Class对象所表示的类或接口所声明的所有字段(包括私有成员),不包括继承属性
    Class.getField(int a)
    返回一个 Field 对象,它反映此Class对象所表示的类或接口的指定公共成员字段
    Class.getFields()
    返回一个包含某些 Field对象的数组,该数组包含此Class对象所表示的类或接口的所有可访问公共字段。

    Q6获取属性的值该怎么办?怎么获取私有属性的值?(提示:参考前面获得私有构造器)
    Q7怎么设置新的属性值?能不能修改私有属性的值?

    2) Field中的方法

    Field.getType():返回这个变量的类型
    Field.getGenericType():如果当前属性有签名属性类型就返回,否则就返回Field.getType()
    Field.getModifiers(): 获取当前成员变量的修饰符。
    Field.get(Obj obj) : 获取属性值
    Field.set(Obj obj,value) :重新设置新的属性值
    注意 :使用反射获取或者修改一个变量的值时,编译器不会进行自动装/拆箱。

    3)使用反射获得和修改属性值

    A7

    package com.testReflect;
    
    import java.lang.reflect.Field;
    import java.lang.reflect.Modifier;
    
    public class FieldTest {
        public static void main(String[] args) throws Exception {
            //使用反射第一步:获取操作类所对应的Class对象
           
            //通过Class类中getField(String name): 获取类特定的方法,name参数指定了属性的名称
    
            //拿到了Field类的实例后就可以调用其中的方法了
    
            //方法:getModifiers()  以整数形式返回由此 Field 对象表示的字段的 Java 语言修饰符
            System.out.println("修饰符:  " + Modifier.toString(field.getModifiers()));
    
            //方法:getType()  返回一个 Class 对象,它标识了此 Field 对象所表示字段的声明类型
            System.out.println("类型:  "+field.getType());
    
            //方法:get(Object obj) 返回指定对象obj上此 Field 表示的字段的值
            System.out.println("属性值:  "+field.get(obj));
    
    
            //方法: set(Object obj, Object value)  将指定对象变量上此 Field 对象表示的字段设置为指定的新值
            
            System.out.println("修改属性值后 --> get(Object obj):  "+field.get(obj));
        }
    }
    

    Q8 请问下列代码使用出错,并说明理由。

    public class FieldA extends A {
        public Integer value;
        public final boolean is = true;
    
        public static void main(String[] args) {
            FieldA fielda = new FieldA();
            Class cls = fielda.getClass();
            Field field = cls.getField("value");
            field.setInt(fielda, 23);
            
            Field fieldis = cls.getDeclaredField("is");
            fieldis.setBoolean(fieldTrouble, false);
        }
    }
    

    A8

    wrong1 : 无法给一个 Integer 类型的变量赋整型值,必须给它赋一个 Integer 对象才可以。
    使用 Field.set(Object obj, Object value) 方法解决这个问题:.set(cls, new Integer(23));
    wrong2 : 这是因为在类初始化后,无形之中会有一个访问限制阻止我们修改 final 类型的变量。由于 Field 继承自 AccessibleObject , 我们可以使用 AccessibleObject.setAccessible() 方法告诉安全机制,这个变量可以访问。
    因此上面的例子中,声明这个变量是可访问的:field.setAccessible(true),运行就正常了。

    Method

    1)Class -> Method

    public Method[] getMethods() : 获取包括自身和继承过来的所有的public方法
    public Method[] getDeclaredMethods() : 获取自身所有的方法(访问权限无关)
    public Method getMethod(String methodName) : 表示调用指定的一个公共的方法(包括继承的)

    2)Method中的方法

    getName():获得完整的方法信息(包括修饰符、返回值、路径、名称、参数、抛出值)
    getModifiers() :获得修饰符
    getReturnType() : 获得返回值
    getExceptionTypes() : 获得异常名称
    methodName() : 表示被调用方法的名字
    parameterTypes() : 表示被调用方法的参数的Class类型如String.class
    getAnnotations() : 获得注解

    3) 反射调用方法

    java.lang.reflect.Method.invoke()方法来反射调用一个方法

    public native Object invoke(Object receiver, Object... args)
            throws IllegalAccessException, IllegalArgumentException, InvocationTargetException;
    
    • 第一个参数 : 方法属于的对象(如果是静态方法,则可以直接传 null
    • 第二个可变参数 : 是该方法的参数

    总结
    对于任意一个类,都能知道这个类的所以属性和方法;对于任何一个对象,都能够调用它的任何一个方法和属性;这样动态获取新的以及动态调用对象方法的功能就叫做反射。

    相关文章

      网友评论

        本文标题:傻瓜教你反射

        本文链接:https://www.haomeiwen.com/subject/fgkmpftx.html