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