之前写了个工具类,用于方便回显数据和保存数据,感兴趣的可以去看我上一篇文章
http://www.jianshu.com/p/c99bb6aa6dc5
今天来介绍一下原理。
大体来说就是遍历Java Bean的成员变量,根据变量名(fieldName)获取到对应的id,进而拿到TextView,然后再给TextView赋值。map与之类似,只是fieldName变成了key。
下面是echoBean的方法代码
public static void echoBean(Object obj, View parent, IdStrConverterForBean idStrConverter) {
//获取bean的运行时类
Class clazz = obj.getClass();
//遍历bean
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
//获取属性名称(String)
String fieldName = field.getName();
//获取值(Object)
Object fieldValueObj = null;
try {
fieldValueObj = field.get(obj);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
//判断,如果是null,直接continue,不回显
if(fieldValueObj == null){
continue;
}
Class fieldClass = field.getType();
//首先判断,如果类型是集合类(Collection或Map),直接continue,不回显
if(Collection.class.isAssignableFrom(fieldClass) || Map.class.isAssignableFrom(fieldClass)){
continue;
}
//其次,如果类型不是基本类型,不是基本类型的封装类,也不是字符串类型,即是Bean,递归
if (!fieldClass.isPrimitive() && !isWrapClass(fieldClass) && !String.class.isAssignableFrom(fieldClass)) {
echoBean(fieldValueObj, parent, idStrConverter);
continue;
}
//否则(是基本类型或字符串的话),直接回显toString
String fieldValueStr = fieldValueObj.toString();
//根据属性名称获取到对应的id字符串
String idStr = idStrConverter.convert(fieldName, clazz);
//根据id字符串,通过反射获取int型id
int integerId = getIntegerId(idStr);
//最后控件赋值,大功告成
TextView tv;
try {
tv = (TextView) parent.findViewById(integerId);
}catch (ClassCastException e){
//转不成TextView,直接continue跳过
e.printStackTrace();
continue;
}
if (tv != null) {
tv.setText(fieldValueStr);
}
}
}
注释写的很清楚,恨不得每一行代码都要介绍一下。总的思想是遍历加递归。IdStrConverterForBean是个接口,里面有个convert方法,用于将fieldName以用户自己的对应关系得到xml布局中的id字符串,比如fieldName为name,布局上的id为tv_name,就可以以convert联系起来。
在下边有一行代码调用到了getIntergerId(String idStr),很明显是利用String类型的id字符串来得到我们最终需要的int型id的,下面看一下这个方法的代码。
//通过反射获取int型的id
private static int getIntegerId(String idStr) {
int integerId = 0;
try {
Field idField = R.id.class.getDeclaredField(idStr);
integerId = idField.getInt(R.id.class);
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
return integerId;
}
对反射熟悉的朋友应该已经明白了。
我们使用id常用的方式就是R.id.xxx,可是它们是怎么实现的呢?现在随便开个项目打开R文件,搜索“class id”,可以看到id是R的静态内部类,而我们用到的所有id都以静态变量的形式存在于id类中。知道这些,我们就要想办法通过字符串获得对应的变量了,参考Class的java文档,通过Class.getDeclaredField(String name)方法可以获取到Field,再参考Field的文档,通过Field.getInt(Object obj)方法就可以获取到我们需要的int型id了。
至于echoMap方法和echoBean思想类似,不再介绍。
与显示对应的是保存,saveAsBean和saveAsMap两个方法也是同样的思想,只不过显示是把Bean(或Map)的值赋给textview,而保存是把textview的值赋给Bean(或Map),也不再介绍,注释中也很明白,感兴趣的可以去github看代码→_→ https://github.com/AnotherJack/EasyEchoDemo
其实原理很简单,仅仅是简单的封装就可以为我们减少大量的重复工作。