Android反射介绍

一、什么是反射

反射机制:允许运行中的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中:

Android反射介绍_第1张图片

            每个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反射介绍)