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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,026评论 19 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,785评论 18 399
  • 文章作者:Tyan博客:noahsnail.com 3.4 Dependencies A typical ente...
    SnailTyan阅读 4,221评论 2 7
  • 一. Java基础部分.................................................
    wy_sure阅读 3,840评论 0 11
  • 在去姑妈家的路上,有几处小水涧,水涧里有石头、有松软透气的沙土,是螃蟹最喜欢的环境。我们搬开大大小小的石头,惊喜地...
    藤木同学阅读 620评论 0 0