JavaAPIAndroid知识Android开发经验谈

Java反射详解

2017-06-15  本文已影响72人  一只好奇的茂

反射(Reflection)能够让运行于 JVM 中的程序检测和修改运行时的行为。

Class类提供了以下方法

Constructor getConstructor(Class[] params)     
// 根据构造函数的参数,返回一个具体的具有public属性的构造函数
Constructor getConstructors()     
// 返回所有具有public属性的构造函数数组
Constructor getDeclaredConstructor(Class[] params)     
// 根据构造函数的参数,返回一个具体的构造函数(不分public和非public属性)
Constructor getDeclaredConstructors()    
// 返回该类中所有的构造函数数组(不分public和非public属性)
Method getMethod(String name, Class[] params)    
// 根据方法名和参数,返回一个具体的具有public属性的方法
Method[] getMethods()    
// 返回所有具有public属性的方法数组
Method getDeclaredMethod(String name, Class[] params)    
// 根据方法名和参数,返回一个具体的方法(不分public和非public属性)
Method[] getDeclaredMethods()    
// 返回该类中的所有的方法数组(不分public和非public属性)
Field getField(String name)    
// 根据变量名,返回一个具体的具有public属性的成员变量
Field[] getFields()    
// 返回具有public属性的成员变量的数组
Field getDeclaredField(String name)    
// 根据变量名,返回一个成员变量(不分public和非public属性)
Field[] getDelcaredFields()    
// 返回所有成员变量组成的数组(不分public和非public属性)

getDeclaredFields 和 getFields 的区别:
getDeclaredFields()获得某个类的所有申明的字段,即包括public、private和proteced,但是不包括父类的申明字段。
getFields()获得某个类的所有的公共(public)的字段,包括父类。

Class<?> superclass = class1.getSuperclass() ;
Class<?>[] interS = class1.getInterfaces() ;
// 获取所有的接口
 Object object = class1.newInstance() ;
 //创建实例化:相当于 new 了一个对象

实例

Person.class

package reflection;
public class Person {
    private String id;
    private String name;
    public String age;
    public static String str = "Json";
    //构造函数1
    public Person() {
        System.out.println( "构造函数  无参" );
    }
    //构造函数2
    public Person(String id) {
        this.id = id;
        System.out.println( "构造函数 id : " + id );
    }
    //构造函数3
    public Person(String id, String name) {
        this.id = id;
        this.name = name;
        System.out.println( "构造函数 id : " + id  + " name: " + name );
    }
    public String getId() {
        return id;
    }
    public void setId(String id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }
    /**
     * 静态方法
     */
    public static String update(int id) {
        return str+id;
    }
}

练习一:通过反射,获取对象实例,并且操作对象的方法

package reflection;
public class T1 {
    public static void main(String[] args) {
        try {
            //创建类
            Class<?> class1 = Class.forName("reflection.Person");
            //创建实例化:相当于 new 了一个对象
            Object object = class1.newInstance();
            //向下转型
            Person person = (Person) object;
            person.setId("100");
            person.setName("jack");
            System.out.println("id: " + person.getId() + " name: " + person.getName());
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

id: 100 name: jack

练习二:通过反射获取对象字段属性,并且赋值

package reflection;
import java.lang.reflect.Field;
public class T1 {
    public static void main(String[] args) {
        try {
            //创建类
            Class<?> class1 = Class.forName("reflection.Person");
            //创建实例
            Object person = class1.newInstance();
            //获得id 属性
            Field idField = class1.getDeclaredField( "id" ) ;
//打破封装  实际上setAccessible是启用和禁用访问安全检查的开关,并不是为true就能访问为false就不能访问
//由于JDK的安全检查耗时较多.所以通过setAccessible(true)的方式关闭安全检查就可以达到提升反射速度的目的
            idField.setAccessible( true );
            //给id 属性赋值
            idField.set(person, "100") ;
            //打印 person 的属性值
            System.out.println( idField.get( person ));
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (SecurityException e) {
            e.printStackTrace() ;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

id 这个属性的是 private 私有的,不能修改它的值。需要添加 idField.setAccessible( true );

练习三:通过反射操作属性和方法

package reflection;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class T1 {
    public static void main(String[] args) {
        try {
            //创建类
            Class<?> class1 = Class.forName("reflection.Person");
            //创建实例
            Object person = class1.newInstance();
            //获得id 属性
            Field idField = class1.getDeclaredField( "id" ) ;
            //打破封装  实际上setAccessible是启用和禁用访问安全检查的开关,并不是为true就能访问为false就不能访问
            //由于JDK的安全检查耗时较多.所以通过setAccessible(true)的方式关闭安全检查就可以达到提升反射速度的目的
            idField.setAccessible( true );
            //给id 属性赋值
            idField.set(  person , "100") ;
            //获取 setName() 方法
            Method setName = class1.getDeclaredMethod( "setName", String.class ) ;
            //打破封装
            setName.setAccessible( true );
            //调用setName 方法。
            setName.invoke( person , "jack" ) ;
            //获取name 字段
            Field nameField = class1.getDeclaredField( "name" ) ;
            //打破封装
            nameField.setAccessible( true );
            //打印 person 的 id 属性值
            String id_ = (String) idField.get( person ) ;
            System.out.println( "id: " + id_ );
            //打印 person 的 name 属性值
            String name_ = ( String)nameField.get( person ) ;
            System.out.println( "name: " + name_ );
            //获取 getName 方法
            Method getName = class1.getDeclaredMethod( "getName" ) ;
            //打破封装
            getName.setAccessible( true );
            //执行getName方法,并且接收返回值
            String name_2 = (String) getName.invoke( person  ) ;
            System.out.println( "name2: " + name_2 );
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (SecurityException e) {
            e.printStackTrace() ;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

id: 100
name: jack
name2: jack

练习四:静态属性、静态方法调用

package reflection;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class T1 {
    public static void main(String[] args) {
        try {
            //创建类
            Class<?> class1 = Class.forName("reflection.Person");
            //获取 nameField 属性
            Field nameField = class1.getDeclaredField( "str" ) ;
            //获取 nameField 的值
            String name_ = (String) nameField.get( nameField ) ;
            //输出值
            System.out.println( name_ );
            //有返回值,有参数
            Method getUpdateMethod = class1.getDeclaredMethod( "update" , int.class ) ;
            String result_4 = (String) getUpdateMethod.invoke( null  , 1 ) ;
            System.out.println( "返回值: "+ result_4 );
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (SecurityException e) {
            e.printStackTrace() ;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

练习五:调用构造函数,创建对象实例

package reflection;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class T1 {
    public static void main(String[] args) {
        try {
            //创建类
            Class<?> class1 = Class.forName("reflection.Person");
            //无参构造函数
            Object object = class1.newInstance() ;
            //有参构造函数:一个参数
            Constructor<?> constructor =  class1.getDeclaredConstructor( String.class ) ;
            constructor.newInstance( "1000" ) ;
            //有参构造函数:二个参数
            Constructor<?> constructor2 =  class1.getDeclaredConstructor( String.class , String.class ) ;
            constructor2.newInstance( "1001" , "jack" ) ;
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (SecurityException e) {
            e.printStackTrace() ;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

参考文章

Java 反射 使用总结

上一篇下一篇

猜你喜欢

热点阅读