「EasyEcho」原理介绍:通过反射获取Android控件id

之前写了个工具类,用于方便回显数据和保存数据,感兴趣的可以去看我上一篇文章

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
其实原理很简单,仅仅是简单的封装就可以为我们减少大量的重复工作。

你可能感兴趣的:(「EasyEcho」原理介绍:通过反射获取Android控件id)