Android自定义Cursor遇到的一个小问题

Android应用很多情况下存储大量数据都会用SQliter,使用sqlliter不免要和cursor打交道,灵活使用cursor会省很多事。如将自己的数据组合在一个虚拟的表中(数据集合),通过Providercursor形式返回给用户,还可以规定cursor每取一次返回多少记录,以减少UI数据量大的负载压力。(如果只是封装下数据以cursor方式返回,MetrixCursor可以完成此需求,其publicvoid addRow (Object[]columnValues)方法可以帮你维护添加进去的对象数组数据)。

具体实现代码如下:

final Map<String, String>  data = new HashMap<String, String>();
        data.put("wifi","on");
        Cursor cursor = new AbstractCursor() {

            private String[] names;

            /**
             * 记录数
             * @return
             */
            @Override
            public int getCount() {
                return data.size() > 0 ? 1 : 0;
            }

            /**
             * 获取每一列的名称,对于SharedPreferences 就是每一个key
             * @return
             */
            @Override
            public String[] getColumnNames() {
                if(names == null) {
                    names = new String[0];

                    if (!data.isEmpty()) {
                        names = new String[data.size()];
                        Set<String> keySet = data.keySet();
                        <span style="color:#ff0000;">names = (String[])keySet.toArray(new String[0]);</span>
                    }
                }

                return names;
            }

            @Override
            public String getString(int column) {
                String key = names[column];

                return (String)data.get(key);
            }

            @Override
            public short getShort(int column) {
                return 0;
            }

            @Override
            public int getInt(int column) {
                return 0;
            }

            @Override
            public long getLong(int column) {
                return 0;
            }

            @Override
            public float getFloat(int column) {
                return 0;
            }

            @Override
            public double getDouble(int column) {
                return 0;
            }

            @Override
            public boolean isNull(int column) {
                return false;
            }
        };
红色部分是要注意的代码,开始是这么写的:

 if(names == null) {
                            names = new String[0];

                            if (!data.isEmpty()) {
                                names = new String[data.size()];
                                Set<String> keySet = data.keySet();
                                int i = 0;
                                for (String key : keySet) {
                                    names[i++] = key;
                                }
                            }
                        }
但是Set有提供的转成Array的方法toArray(),于是改成了
 <span style="color:#ff0000;">names = (String[])keySet.toArray();</span>
但是结果是错误的,打断点跟踪后发现cursor中是有数据的,但是如果取得花是会出错的。

对于Set而言,它只知道它内部保存的是Object,所以默认情况下,toArray只能是返回一个由这些Object构成的Object数组出来。
但程序的作者或许更清楚其内部元素的更具体的类型,因此,Set类提供了toArray的另一个重载版本,允许用户指定一种比Object[]更具体的数组类型,方法是传递一个用户想要的数组类型的一个数组实例进去,多长都无所谓(因此我们常常使用一个0长度的,毕竟把类型带进去就OK了),于是,toArray内部就会按照你想要的这种类型,给构造一个数组出来。这样构造出来的数组,当然是很安全地被调用者转换回那个实际的类型。

参考:http://bbs.csdn.net/topics/100149308

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。