Android应用很多情况下存储大量数据都会用SQliter,使用sqlliter不免要和cursor打交道,灵活使用cursor会省很多事。如将自己的数据组合在一个虚拟的表中(数据集合),通过Provider以cursor形式返回给用户,还可以规定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