美文网首页
Android反射介绍

Android反射介绍

作者: viky_lyn | 来源:发表于2018-04-16 11:21 被阅读0次

    一、什么是反射

    反射机制:允许运行中的Java程序对自身进行检查,并能直接操作程序的内部属性或方法。允许程序在正在执行的过程中,利用Reflection APIs取得任何已知名称的类的内部信息,包括:package、 type parameters、 superclass、 implemented interfaces、 inner classes、 outer classes、 fields、 constructors、 methods、 modifiers等,并可以在执行的过程中,动态生成Instances、变更fields内容或唤起methods。

    简单来说,我们可以利用反射机制在Java程序中,动态的去调用一些protected甚至是private的方法或类。

    二、为什么要用反射

    1、Android SDK的源码中,很多类或方法中经常加上了“@hide”注释标记,它的作用是使这个方法或类在生成SDK时不可见,那么我们的程序可能无法编译通过,而且在最终发布的时候,就可能存在一些问题,利用反射,就可以获取到该方法或类,从而使编译通过并可正常使用该方法或类。

           2、Android的项目可以分为不同的module,例如App中:

                每个module只能调用自己内部的方法,若在Amodule中想调用Bmodule里的方法,则可以采用:

                 a. 在Amodule里的build.gradle中的dependencies节点内,添加compile project(':debugTool')内容,其中“debugTool”为Bmodule的模块名。

                 b. 或在Amodule中,利用反射获取Bmodule里的类、方法及变量。

            3. 举例说明

               在上图中,提bug的SDK位于debugTool下,而调用SDK的方法,在common中,即SDK所在位置和调用SDK方法的位置,不在同一个module中,所以无法直接调用SDK内容。

               考虑到:i. 不想入侵原有的代码过多

                             ii. 也不想改变原有代码的结构

                             iii. 需要调用SDK的内容也不是很多

                             iv. 我对SDK内部很熟悉,对将来的改动也能了如指掌。

               所以选择上述2中的b方法,采用反射方式调用SDK中的内容。

    三、怎么实现反射

    3.1、首先,此处给出反射相关的API文档链接:java.lang.reflect

           3.2、接下来,举例说几个常用的方法:

                 1、 找到指定的类(包括class、interface、service等)

                       用法:Class.forName("包名.类名")。例如:

    final Class c = Class.forName("xxx.createjirabug.bugEntrance.FxService");、

    2、获取指定类中的指定方法

                       用法:c.getMethod("方法名","参数类型")。例如:

    final Class c = Class.forName("xxx.createjirabug.bugEntrance.FxService");

                                  Method setListener = c.getMethod("setListener", View.OnClickListener.class);

    3、 执行指定的方法

    用法:method.invoke(receiver, "具体参数")。例如:

    Method setListener = c.getMethod("setListener", View.OnClickListener.class);

    setListener.invoke(o, new View.OnClickListener() {

    @Override

    public void onClick(View v) {

    });

                 4、 判断classA是否为classB的子类

                       用法:boolean is = classA.isInstance(classB)。例如:

    Class rnBase = Class.forName("xxx.reactnative.RnBaseActivity");

                                  xxxBaseFragmentActivity currentxxxBaseActivity = xxxActivityStack.getInstance().getCurrentxxxBaseActivity();

                                  if(rnBase.isInstance(currentxxxBaseActivity)){

                                  }

                 5、 获取指定的变量

                       用法:class.getField("变量名")。例如:

    Class buildConfig = Class.forName("xxx.BuildConfig");//先获取buildConfig类

    Field tField =buildConfig.getField("IS_CONTAIN_BONREE");//再从类中获取IS_CONTAIN_BONREE变量

    tField.setAccessible(true);//buildConfig类中的IS_CONTAIN_BONREE成员变量为private,故必须进行此操作,忽略访问权限。

    booleanb = tField.getBoolean(null);//获取tField的布尔值(我已知该变量为布尔类型)

     6、以上是反射中常用的方法,若需要用到的方法不在以上提供的内容中,请查阅3.1给出的api文档,查阅具体资料。

    四、为什么反射不起作用

           将一大串的反射写完后,对于新手来说,经常遇到写完的反射不起作用的情况,查阅log又没有发现crash内容,此时该怎么办呢?

           请按照以下步骤查看具体出错原因:

           1、所有的反射代码都会被要求写入try...cach中:

                     若未找到class,会抛出ClassNotFoundException异常;

                     若未找到method,会抛出NoSuchMethodException异常;

                     若未找到field,会抛出NoSuchFieldException异常;

                     若方法执行(invoke)出错,会抛出InvocationTargetException异常;

    若因访问权限被禁止,会抛出IllegalAccessException异常;

                     ......

     请将cach住的所有异常打印出来,查阅具体原因:

                     class未找到:是否class的访问权限问题?是否class的包名或类名写错?是否有该类?

                     method未找到:是否method的访问权限问题?是否method的名称写错?是否method的参数类型及参数个数有误?是否reciever传入有误?是否有该方法?

                     field未找到:是否field的访问权限问题?是否field的名称写错?是否field的类型有误?是否有该变量?

                     ......

               若以上方法尝试过后,发现都没有异常情况,请查看步骤2。

           2、当前项目是否涉及到混淆?打包的时候,代码是否被混淆?SDK中需要被反射的类、方法、变量是否被混淆了?

    如果是,请在Bmodule的proguard-project.txt文件中,添加SDK中相关的包,进行避混处理。

    相关文章

      网友评论

          本文标题:Android反射介绍

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