android.database.CursorIndexOutOfBoundsException

Posted

技术标签:

【中文标题】android.database.CursorIndexOutOfBoundsException【英文标题】: 【发布时间】:2013-11-27 06:34:22 【问题描述】:

我的 android 应用中的数据访问代码引发了异常。

这是导致异常的代码:

String getPost = "SELECT * FROM " + TABLE_POST + ";";
Cursor result = getReadableDatabase().rawQuery(getPost, null);
List posts = new ArrayList();
Log.d("count", result.getCount() + " post rows");
while (result != null && result.moveToNext());
    Post post = new Post();
    post.setPostId(result.getInt(0));
    posts.add(post);
    ....

引发的异常是:

: Index 2 requested, with a size of 2 exception

log 方法产生结果2 post rows

这里有什么问题?

编辑: 我已经添加了完整的堆栈跟踪,以便那些希望查看它的人可以这样做。

11-14 09:57:50.538: W/dalvikvm(4710): threadid=1: thread exiting with uncaught exception (group=0x40a71930)
11-14 09:57:50.608: E/AndroidRuntime(4710): FATAL EXCEPTION: main
11-14 09:57:50.608: E/AndroidRuntime(4710): java.lang.RuntimeException: Unable to start activity ComponentInfocom.innolabmm.software.collaboration/com.innolabmm.software.collaboration.PostCommentActivity: : Index 2 requested, with a size of 2
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2180)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2230)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread.access$600(ActivityThread.java:141)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1234)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.os.Handler.dispatchMessage(Handler.java:99)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.os.Looper.loop(Looper.java:137)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread.main(ActivityThread.java:5041)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at java.lang.reflect.Method.invokeNative(Native Method)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at java.lang.reflect.Method.invoke(Method.java:511)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at dalvik.system.NativeStart.main(Native Method)
11-14 09:57:50.608: E/AndroidRuntime(4710): Caused by: : Index 2 requested, with a size of 2
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.database.AbstractCursor.checkPosition(AbstractCursor.java:424)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.database.AbstractWindowedCursor.checkPosition(AbstractWindowedCursor.java:136)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.database.AbstractWindowedCursor.getInt(AbstractWindowedCursor.java:68)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at com.innolabmm.software.collaboration.data.CollaborationDbHandler.fetchAllPost(CollaborationDbHandler.java:362)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at com.innolabmm.software.collaboration.PostCommentActivity.onCreate(PostCommentActivity.java:48)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.Activity.performCreate(Activity.java:5104)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1080)
11-14 09:57:50.608: E/AndroidRuntime(4710):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2144)
11-14 09:57:50.608: E/AndroidRuntime(4710):     ... 11 more

【问题讨论】:

请显示完整的堆栈跟踪,而不仅仅是类型和消息。 您还没有向我们展示所有代码。您在某处调用 result.getInt(2)result.getString(2) 或使用索引 2。 我已经添加了堆栈跟踪,但我认为问题已经解决了。 【参考方案1】:

您正试图检索索引 2 上的项目,但该索引确实不存在(光标大小为 2,因此索引为 0,1)。

改变你的循环:

if (result != null && result.moveToFirst())
    do 
       Post post = new Post();
       post.setPostId(result.getInt(0));
       posts.add(post);
       ....
     while (result.moveToNext());

现在它应该可以正常工作了。

注意:不要忘记调用moveToFirst() 方法将光标移动到第一条记录(隐式位于第一行之前)并准备读取。这也是测试 Cursor 是否有效的便捷方法。

注意2:不要使用列索引,你只会在计数时出错。而不是使用列名 - 通常建议使用这种方法,例如cursor.getColumnIndex("<columnName>")

【讨论】:

非常感谢您的回答。它应该会改进我从现在开始编写的 Android 数据访问代码。 我的光标对象正在播放,您的解决方案已修复它!非常感谢!【参考方案2】:

你还没有将光标索引移到第一个.. 试试下面这样

 String getPost = "SELECT * FROM " + TABLE_POST + ";";
    Cursor result = getReadableDatabase().rawQuery(getPost, null);
    List posts = new ArrayList();
    Log.d("count", result.getCount() + " post rows");

    if (result .moveToFirst()) 
        do 
             Post post = new Post();
             post.setPostId(result.getInt(0));
             posts.add(post);
              ....
         while (result .moveToNext());
    
    result .close();

【讨论】:

【参考方案3】:

如果我们的光标有数据但尚未指向数据列表中的第一个索引,就会出现这种情况。

为了避免这个异常,我们首先应该检查:

if (!cursor.moveToFirst())
        cursor.moveToFirst();

然后进行进一步的操作。

【讨论】:

也许我遗漏了什么,但为什么要调用 cursor.moveToFirst() 两次? @BrillPappin 你只调用了一次。 if (!cursor.moveToFirst()) 只是检查它是否已经移动。并且cursor.moveToFirst(); 仅在它没有移动到第一个时才被调用。 不,您实际上是在调用它两次。为您提供结果所做的工作是相同的。因此 cursor.moveToFirst() 在 IF 条件内不会改变其功能。您不需要第二次调用,因为 IF 块中的第一次调用将完成您需要的所有操作(显然,不要反转结果)。 自己试一试,写个单元测试,双向尝试。

以上是关于android.database.CursorIndexOutOfBoundsException的主要内容,如果未能解决你的问题,请参考以下文章