随着工作年限的增加,日常工作中的需求已不再是findViewById那么简单了,更多的倾向于框架化,源码化的方向发展,什么叫源码化呢?就是我们在做一个工具(比如一个像自定义IOC工具)的时候更多的像源码发起进攻,分析他然后掌握它,并且在做这些工具时往往有些系统类或类中的方法被隐藏,这是就需要使用反射来解决。现在来例举下常用的反射操作。
一、是什么
反射是什么,许多博客上面都把书上的解释拿过来照着打,但是往往书上的理论性文字很难理解,所以这里概括一句话:“反射就是获取你想获取的一个类中的类实例、方法、成员变量。即使这个类不对外暴露。”
二、举例
先创建一个用于演示的JavaBean
public class Test {
private String code;
private String msg;
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
}
2.1、获取类实例
//假如Test是一个被hind的系统类,一般方法实例化不了。so?用反射实例化
Test test = Test.class.newInstance();
2.2、获取单个方法并调用
//单个方法
//先向反射获取的类实例调用setCode对code成员变量赋值
Method method = Test.class.getDeclaredMethod("setCode", String.class);
method.setAccessible(true);
//不管是什么是有权限设置比如private什么的,都可以获取,建议不管怎么样都设置下,这样就不会忘
method.invoke(test,"3333");//调用方法
//向反射获取的类实例调用getCode对code成员变量取值
Method method1 = Test.class.getDeclaredMethod("getCode");
method1.setAccessible(true);
//不管是什么是有权限设置比如private什么的,都可以获取,建议不管怎么样都设置下,这样就不会忘
String s = (String) method1.invoke(test);//执行单个方法
System.out.println("yanjin-----"+s);
网上很多人问getDeclaredMethod与getMethod有什么区别,我觉得记住getDeclaredMethod能获取私有化方法就行就算被private修饰。但是记得拿到方法后设置setAccessible(true);不然也只能获取public的方法。getDeclaredMethod的调用参数,第一个是方法名比如上面的"setCode",第二个是参数类型的clazz,参数是String类型的就传入String.class,无参就不用管啦。invoke方法第一个参数是实例化的对象,如果有参数就要传入具体值。
2.3获取单个属性并调用
//单个属性
//获取某个属性的反射对象
Field field = Test.class.getDeclaredField("msg");
field.setAccessible(true);
//获取属性名字
String name = field.getName();
//获取属性类型
String type = field.getType().getName();
//设置属性
field.set(test,"hahhaha");
//获取属性,
Object values = field.get(test);
这里没什么要注意的,谈的上的也有field.setAccessible(true);不要忘了。另外field.get(test);返回的是一个obj类型的,但是field里面也有getInt等方法,所以需要将这个obj转为Javabean中对应的类型,这里需要封装,基于上面已经获取的类型type,做一个if判断即可,这里提供个思路。
//默认返回的是obj,那么如果转换成对应属性,可以自己封装,比如
String s1 =null;
int anInt=-1;
if (type.contains("String")) {
s1 = field.get(test) + "";
} else if (type.contains("int")) {
anInt = field.getInt(test);
}
2.4获取所有成员变量或方法
//获取所有方法
Method[] methods = Test.class.getDeclaredMethods();
for (Method method : methods) {
method.setAccessible(true);
......
}
//获取所有属性
Field[] fields = Test.class.getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
......
}
获取method或field 的单一实例后,之后的操作就和之前的操作一样了。
三、总结
本文介绍的反射用法并不是很全,有些东西也没涉及到,比如:asSubclass的用法
Class.forName("java.util.ArrayList").asSubclass(List.class).newInstance();//如果forName里面填写的类是asSubclass里参数的子类,就实例化他
上述是写IOC注解时用到的,并不是其他人那种直接copy。所以,希望不喜勿喷。