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的主要内容,如果未能解决你的问题,请参考以下文章