android.database.CursorWindowAllocationException: 关闭游标后,2048 kb的游标窗口分配失败

15

在 Stack Overflow 上有许多关于 CursorWindowAllocationException 的问题:

它们都建议使用后必须关闭游标。但这并没有解决我的问题。这是我的代码:

String query = "select serial from tbl1 union select serial from tbl2 union select serial from tbl3";
    SQLiteDatabase db = null;
    Cursor cur = null;
    try {
        SettingsDatabaseHelper dal = new SettingsDatabaseHelper(
                c);
        db = dal.getReadableDatabase();
        cur = db.rawQuery(query, null);
        int numRows = cur.getCount();
        if (numRows > 0) {
            cur.moveToFirst();

            int serialIdx = cur.getColumnIndexOrThrow("serial");
            for (boolean hasItem = cur.moveToFirst(); hasItem; hasItem = cur
                    .moveToNext()) {

                String serial = cur.getString(serialIdx);
                if (Validator.getInstance().isValidSerial(serial))
                    serials.add(serial);

            }
        }
    } finally {
        if (cur != null)
            cur.close();

        if (db != null)
            db.close();
    }

我每隔几秒钟运行一次这个方法。半小时后,我在int numRows=cur.getCount();处遇到了CursorWindowAllocationException,然后我的服务停止了。

由于我已经关闭了游标,可能是我的Helper类出了问题。这是代码:

public class SettingsDatabaseHelper extends SQLiteOpenHelper {

    private static final String DATABASE_NAME = "mydb.db";
    private static final int DATABASE_VERSION = 1;


    private static final String TBL_CREATE="create table...";



    public SettingsDatabaseHelper(Context context) {
        super(context, DATABASE_NAME, null, DATABASE_VERSION);
    }

    @Override
    public void onCreate(SQLiteDatabase database) {
        database.execSQL(TBL_CREATE);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        onCreate(db);
    }
}

我该如何避免抛出这个异常?

3个回答

1
我正在从一个服务中查询数据库。当我从活动中进行查询时,问题再也没有出现过。
除了查询之外,我还在同一个服务中从串口读取数据。也许这是导致问题的原因。然而,当我使用活动代替服务时,问题再也没有发生过。

2
我没有得到答案。我有同样的问题。 - Nitin Karale

0

我曾经遇到同样的问题,在服务类中使用无限循环调用数据库。

我的代码像这样:

service ()
 {
void run()
   {
     while(conditionisTrue)
       {
              db.getRecord();  //I remove this code from here and put it in the start of service c
        }
    }
 }

0

在我的应用程序中,当从Android媒体库加载超过500首歌曲时,我遇到了同样的问题。

关闭游标并不能解决你的问题。

你应该使用AsyncTask。在doInBackground方法中编写从游标检索值的代码。

public class CursorLoader extends AsyncTask {

    @Override
    protected Void doInBackground(Void... params) {

        String query = "select serial from tbl1 union select serial from tbl2 union select serial from tbl3";
SQLiteDatabase db = null;
Cursor cur = null;
try {
    SettingsDatabaseHelper dal = new SettingsDatabaseHelper(
            c);
    db = dal.getReadableDatabase();
    cur = db.rawQuery(query, null);
    int numRows = cur.getCount();
    if (numRows > 0) {
        cur.moveToFirst();

        int serialIdx = cur.getColumnIndexOrThrow("serial");
        for (boolean hasItem = cur.moveToFirst(); hasItem; hasItem = cur
                .moveToNext()) {

            String serial = cur.getString(serialIdx);
            if (Validator.getInstance().isValidSerial(serial))
                serials.add(serial);

        }
    }
} finally {
    if (cur != null)
        cur.close();

    if (db != null)
        db.close();
}

        return null;
    }

    @Override
    protected void onPreExecute() {

    }



    @Override
    protected void onPostExecute(Void result) {


}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接