Cursor.moveToNext错误

时间:2016-10-23 12:59:41

标签: java android sqlite android-contentprovider android-cursor

我偶尔会看到一份崩溃报告:

Fatal Exception: java.lang.IllegalStateException: Couldn't read row 1127, col 0 from CursorWindow.  Make sure the Cursor is initialized correctly before accessing data from it.
   at android.database.CursorWindow.nativeGetLong(CursorWindow.java)
   at android.database.CursorWindow.getLong(CursorWindow.java:511)
   at android.database.AbstractWindowedCursor.getLong(AbstractWindowedCursor.java:75)
   at android.database.AbstractCursor.moveToPosition(AbstractCursor.java:220)
   at android.database.AbstractCursor.moveToNext(AbstractCursor.java:245)
   at android.database.CursorWrapper.moveToNext(CursorWrapper.java:166)
   at com.anthonymandra.util.ImageUtils.cleanDatabase(SourceFile:381)

显然moveToNext在循环中失败(注意第1127行)。循环删除表示无法再找到的文件的条目。

final ArrayList<ContentProviderOperation> operations = new ArrayList<>();

try( Cursor cursor = c.getContentResolver().query(Meta.CONTENT_URI, null, null, null, null))
{
    if (cursor == null)
        return;

    final int uriColumn = cursor.getColumnIndex(Meta.URI);
    final int idColumn = cursor.getColumnIndex(BaseColumns._ID);

    while (cursor.moveToNext())
    {
        String uriString = cursor.getString(uriColumn);
        if (uriString == null)  // we've got some bogus data, just remove
        {
            operations.add(ContentProviderOperation.newDelete(
                    Uri.withAppendedPath(Meta.CONTENT_URI, cursor.getString(idColumn))).build());
            continue;
        }
        Uri uri = Uri.parse(uriString);
        UsefulDocumentFile file = UsefulDocumentFile.fromUri(c, uri);
        if (!file.exists())
        {
            operations.add(ContentProviderOperation.newDelete(Meta.CONTENT_URI)
                    .withSelection(getWhere(), new String[]{uriString}).build());
        }
    }
}

c.getContentResolver().applyBatch(Meta.AUTHORITY, operations);

知道游标如何在循环中失败吗?

2 个答案:

答案 0 :(得分:4)

您似乎正在进行一个相当大的查询:至少1127行,以及所有可能的列(尽管您只使用其中两个)。并且,在您使用Cursor工作期间,假设ContentProvider与Android的UsefulDocumentFile相关,您正在将磁盘I / O和/或IPC重新发送回DocumentFile

正如Prakash所说,你得到的Cursor可能只包含一部分信息。只要您尝试超越该点,Cursor就需要返回数据源并获取下一个结果窗口。如果在进行此项工作时数据发生了重大变化(例如,现在的行数少于1127行),我可以看到您遇到此类问题。

我建议你:

  • 限制您返回所需子集的列,

  • 在循环期间避免I / O(例如,旋转Cursor以构建ArrayList<Pair>或其他内容,关闭Cursor,然后遍历列表)

答案 1 :(得分:3)

可能是您在数据库中写了文件,这就是您获得java.lang.IllegalStateException

的原因
 UsefulDocumentFile file = UsefulDocumentFile.fromUri(c, uri);

Android SQLite返回光标窗口中的行,其最大大小为2MB config_cursorWindowSize。如果您的行超出此限制,您将收到此错误。

将文件存储在数据库中的文件系统和路径中。