一、什么是反射
反射机制:允许运行中的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中相关的包,进行避混处理。