JAVA

Java基础(三)-反射

2019-11-05  本文已影响0人  Stan_Z
一、概念

动态获取对象信息和调用对象方法的功能称之为反射。

二、反射的方式

Class.forName() 得到的Class是经过加载、连接和初始化。
Classloder.loadClass() 得到的Class是只进行加载 ,并没有进行连接和初始化。
因此需要初始化的选择Class.forName(),例如数据库连接,在静态块中注册了driver,因此需要初始化。

三、基本使用
public class Person {
   public String name = "Stan";
   private static int age = 3;
   private String gender = "man";
   public Person(){
       System.out.println("construction method");
   }
   public String getName(){
       return name;
   }
   private static int getAge(){
       return age;
   }
   private String modifyInfo(String name){
       this.name = name;
       return name;
   }
}

public static void main(String[] args) {
    try {
       //获取对应的Class
       String className = Person.class.getName();
       Class clz = Class.forName(className);
       //根据双亲委派原则来加载Class.
       Class clz = Reflection.class.getClassLoader().loadClass(className);
       Class clz = ClassLoader.getSystemClassLoader().loadClass(className);
       //直接通过类来获取对应的Class
       Class clz = Person.class;
       //这几种写法都OK
    } catch (Exception e) {
            e.printStackTrace();
        }
}

//公有属性
public static void getPublicField(Class clz) throws Exception{
        Person obj = (Person) clz.newInstance();//创建对象,前提是没有显式构造方法。
       //创建对象后可以直接获取其public的属性和方法
        System.out.println(obj.name);
        System.out.println(obj.getName());
}

//获取带参的公有构造方法
public static void getPublicStructor(Class clz)throws Exception{
        Constructor publicConstructor = clz.getDeclaredConstructor(String.class);//先获取构造方法并传参
        Person obj = (Person) publicConstructor.newInstance("Milo”);//然后创建对象并赋值,这也是有显示构造方法的初始化方式
        System.out.println(obj.name); //同样可以去获取其public的属性和方法
        System.out.println(obj.getName());
}

//获取私有属性
//getDeclaredFields() 方法只能获取本类中定义的所有方法(public/protected/default/private),获取不到父类中定义的方法,即使是product的方法也获取不到
//getFields() 方法能获取到本类及父类的方法,但是这些方法必须是public的,其他的都获取不到。
public static void getPrivateField(Class clz)throws Exception{
        Object o = clz.newInstance();
       //clz.getDeclaredFields() 获取所有属性 
        Field ageField = clz.getDeclaredField("age”);//获取单个属性
        ageField.setAccessible(true);//私有方法需求通过该方法绕过权限
        ageField.setInt(o,10);//设置属性
        Integer age = (Integer)ageField.get(o);//获取属性
        System.out.println(age);
}

//获取带参的私有方法
public static void getPrivateMethod(Class clz) throws Exception{
        Object o = clz.newInstance();
       //clz.getDeclaredMethods()获取所有方法
        Method modifyInfoMethod = clz.getDeclaredMethod("modifyInfo", String.class);//获取当个方法,并传参
        modifyInfoMethod.setAccessible(true);
        String result = (String) modifyInfoMethod.invoke(o, "Mark”);//方法使用
        System.out.println(result);
}

//获取属性类型:field.getGenericType().toString()
//获取属性修饰符(public/private等) field.getModifiers()

非常基础的使用,但是已经很覆盖到大多数的场景了。

四、@hide能否反射获取

这部分还没细致研究,看了一哥们的文章,可信度比较高,后续可以验证下
android9.0后对hide方法反射限制的分析

五、反射的性能问题

Java反射到底慢在哪?
这篇文章有具体的实验测试数据,可信度比较高。从结论来看:耗时方法主要是getMethod、getDeclaredField、invoke和set四个方法,最终调用的都是native方法,猜想有可能是jni调用 外加各种判断逻辑耗时。另外,反射导致的性能问题是否严重跟使用的次数有关系,如果控制在100次以内,基本上没什么差别,如果调用次数超过了100次,性能差异会很明显。

上一篇 下一篇

猜你喜欢

热点阅读