当前位置: 代码迷 >> Android >> Android自定义Cursor遇到的一个小疑点
  详细解决方案

Android自定义Cursor遇到的一个小疑点

热度:59   发布时间:2016-04-28 01:06:45.0
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();                        names = (String[])keySet.toArray(new String[0]);                    }                }                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;            }        };
第28行代码是要注意的代码,开始是这么写的:

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(),于是改成了
names = (String[])keySet.toArray();
但是结果是错误的,打断点跟踪后发现cursor中是有数据的,但是如果取结果是会出错的。

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

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

  相关解决方案