Java反射机制

2017-06-14  本文已影响0人  ReturnYHH

前言

反射,相信大家都听说过,又或者在项目中使用过,做过插件开发的同学应该对它并不陌生,我们需要调用另外一个apk的资源,就需要使用反射去获取,现在就来聊聊反射


什么是反射

根据文档介绍:

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

顾名思义,它的强大在于动态获取信息,动态调用对象,通俗来讲,就是它可以加载一个运行时才得知名称的class,并且获得改class的构造体,从而生成完整的对象,为什么需要这样做?做过插件开发的同学应该知道,我们的插件apk都是动态加载的,需要时才用,而且插件apk和我们的主apk并不是同一个,插件apk可能存放在sd卡中,也可能存放在assets中,既然我们是动态加载,那么当插件apk运行起来的时候,我们就可以通过反射机制去获得它的信息,调用它的对象,那么,它是怎么工作的以及是怎么使用的?接下来我们看看它的实现过程


反射使用

首先我们创建一个对象UserBean,里面存放几个字段:

public class UserBean {

    private String id;
    private String 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;
    }

    @Override
    public String toString() {
        return "UserBean{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}

然后,我们来看看怎样获取反射类,并且获取到对象的构造方法

        // 获取Class对象
        // 方式一
        Class clazz= Class.forName("全类名");
        // 方式二
        Class<UserBean> clazz = UserBean.class;
        // 方式三
        UserBean user = new UserBean();
        Class clazz = user.getClass();
 try {
            Class<?> clazz = Class.forName("com.designpattern.myplugdemo.UserBean");
            //获取该类中所有的public修饰的构造方法
//            Constructor<?>[] constructors = clazz.getConstructors();
            //获取该类中无参的public修饰的构造方法
            Constructor constructor = clazz.getConstructor();

            Log.e("tag",constructor.toString()+"");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        }
    }

相关的获取类的构造方法的方法有

    // 返回 Class 对象所表示的类的系统的无参构造方法
    Constructor<T> getConstructor()
    // 返回 Class 对象所表示的类的指定public的构造方法
    Constructor<T> getConstructor(Class<?>... parameterTypes)
    // 返回Class 对象所表示的类的所有public的构造方法数组
    Constructor<?>[] getConstructors()
    // 返回Class 对象所表示的类或接口的指定构造方法
    Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
    // 返回 Class 对象表示的类声明的所有构造方法数组(可以是非public的)
    Constructor<?>[] getDeclaredConstructors()

但是注意了,如果使用的是getConstructor(),那么就算你写了多个有参的构造方法,也不会返回,它只会返回系统默认的无参的构造方法,如果我们没有指定返回的构造方法,默认也是返回系统的无参的构造方法,我们来看下完整的代码

 try {
            Class<?> clazz = Class.forName("com.designpattern.myplugdemo.UserBean");
            //实力化对象
            UserBean userBean = (UserBean) clazz.newInstance();
            userBean.setId("1");
            userBean.setName("test");

            Log.e("tag",userBean.toString());
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

如果我们需要指定构造方法,或者说对于多个构造方法进行不同的赋值,需要在bean中添加不同的构造方法,然后可以这样做


指定构造方法

首先我们在bean中添加一个两个参数的构造方法

 public UserBean(String id, String name){
        this.id=id;
        this.name=name;
    }

然后外部调用代码

try {
            Class<?> clazz = Class.forName("com.designpattern.myplugdemo.UserBean");
            Constructor<?> constructor = clazz.getDeclaredConstructor(String.class, String.class);
            UserBean userBean = (UserBean) constructor.newInstance("2","test");

            Log.e("tag",userBean.toString());
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

多个构造方法

我们在bean中添加两个构造方法

  public UserBean(String id){
        this.id=id;
    }
    public UserBean(String id, String name){
        this.id=id;
        this.name=name;
    }

然后外部调用

try {
            Class<?> clazz = Class.forName("com.designpattern.myplugdemo.UserBean");
            Constructor<?>[] constructors = clazz.getConstructors();
            UserBean userBean1 = (UserBean) constructors[0].newInstance("1");
            UserBean userBean2 = (UserBean) constructors[1].newInstance("1","test");

            Log.e("userBean1", userBean1.toString());
            Log.e("userBean2", userBean2.toString());

        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

这里简单介绍反射的运用,还有很多其他的用法,这里就不一一详细了,有兴趣的话可以查看下其他资料,end

上一篇下一篇

猜你喜欢

热点阅读