Kotlin数据库错误CursorIndexOutOfBoundsException
这是错误的,它说索引超出界限,但我不能如何解决它,有一些土耳其语的话,但他们并不重要,我认为:
E/AndroidRuntime: FATAL EXCEPTION: main Process: com.example.burhanozen.nothesaplama, PID: 26919 java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.burhanozen.nothesaplama/com.example.burhanozen.nothesaplama.MainActivity}: android.database.CursorIndexOutOfBoundsException: Index 1 requested, with a size of 1 at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2665) at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2726) at ........
我试图存储某种学生信息。 我有这些错误,我在下面分享代码。 这是我的MainActivity:
override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) val toolbar = findViewById(R.id.toolbar) as Toolbar setSupportActionBar(toolbar) val fab = findViewById(R.id.fab) as FloatingActionButton fab.setOnClickListener { val intent = Intent(this@MainActivity,NotEkrani::class.java) startActivity(intent) } val studentsArray = ArrayList<String>() val arrayAdapter = ArrayAdapter(this,android.R.layout.simple_list_item_1,studentsArray) listView.adapter= arrayAdapter try{ val myDatabase = this.openOrCreateDatabase("student", Context.MODE_PRIVATE,null) myDatabase.execSQL("CREATE TABLE IF NOT EXISTS students (isim TEXT," + " dersbir TEXT, dersbirkredi TEXT, dersbirort TEXT," + " dersiki TEXT, dersikikredi TEXT, dersikiort TEXT) ") val cursor = myDatabase.rawQuery("select * from students",null) val nameIx = cursor.getColumnIndex("isim") val dersbirIx = cursor.getColumnIndex("dersbir") val dersbirkrediIx = cursor.getColumnIndex("dersbirkredi") val dersbirortIx = cursor.getColumnIndex("dersbirort") val dersikiIx = cursor.getColumnIndex("dersiki") val dersikikrediIx = cursor.getColumnIndex("dersikikredi") val dersikiortIx = cursor.getColumnIndex("dersikiort") cursor.moveToNext() while (cursor != null){ studentsArray.add(cursor.getString(nameIx)) studentsArray.add(cursor.getString(dersbirIx)) studentsArray.add(cursor.getString(dersbirkrediIx)) studentsArray.add(cursor.getString(dersbirortIx)) studentsArray.add(cursor.getString(dersikiIx)) studentsArray.add(cursor.getString(dersikikrediIx)) studentsArray.add(cursor.getString(dersikiortIx)) cursor.moveToNext() arrayAdapter.notifyDataSetChanged() } while (cursor!=null){ cursor!!.close() } }catch (e:SQLException){ }
这是我的第二个活动页面,我不能在这里任何错误,但可能是这可以帮助。 我在这里写了数据库代码。
fun kaydet(view:View){ val isim = isim.text.toString() val dersbir = dersbir.text.toString() val dersbirkredi = dersbirkredi.text.toString() val dersbirort = dersbirort.text.toString() val dersiki = dersiki.text.toString() val dersikikredi = dersikikredi.text.toString() val dersikiort = dersikiort.text.toString() try { val myDatabase = openOrCreateDatabase("student", Context.MODE_PRIVATE,null) myDatabase.execSQL("CREATE TABLE IF NOT EXISTS students (isim TEXT," + " dersbir TEXT, dersbirkredi TEXT, dersbirort TEXT," + " dersiki TEXT, dersikikredi TEXT, dersikiort TEXT) ") val sqlString = "INSERT INTO students (isim,dersbir,dersbirkredi,dersbirort,dersiki,dersikikredi,dersikiort) VALUES(?,?,?,?,?,?,?)" val statement = myDatabase.compileStatement(sqlString) statement.bindString(1,isim) statement.bindString(2,dersbir) statement.bindString(3,dersbirkredi) statement.bindString(4,dersbirort) statement.bindString(5,dersiki) statement.bindString(6,dersikikredi) statement.bindString(7,dersikiort) statement.execute() }catch (e:SQLException){ } val intent = Intent(this,MainActivity::class.java) startActivity(intent) }
改变你的while循环看起来像这样:
while (cursor.moveToNext()){ studentsArray.add(cursor.getString(nameIx)) ... }
(从您的代码中删除所有其他cursor.moveToNext()
)
您正在while循环条件中检查cursor != null
,但在读取游标中的所有行之后,它不会变为空。
在上面的代码中,如果下一行有cursor.moveToNext
返回true。 否则返回false,循环终止。
而且,刚刚注意到,你应该改变这个,
while (cursor!=null){ cursor!!.close() }
到一个if检查:
if (cursor!=null){ cursor.close() }
或者,正如Todd在评论中所建议的,您可以使用自动关闭游标的高阶函数use
,这与Java 7的try-with-resources类似。
cursor.use { while (cursor.moveToNext()){ studentsArray.add(cursor.getString(nameIx)) ... } }
你得到这个错误,因为你的光标大小是1.数据库中只有一条记录。 但是您正在查询数据库超过一次。 这是你的错误的原因。
首先你的完整的执行代码是错误的。
您正在您的活动中创建数据库。 如果你这样做,你不能正确和有效地管理你的数据库。
首先,你必须在一个单独的类中创建一个SqliteOpenHelper数据库。 那么你必须在那里创建一个表。 然后在那里编写帮助器方法来从中获取和插入值。
你的数据库类应该像下面的例子。
class DbHelper(context: Context) : SQLiteOpenHelper(context, "example.db", null, 4) { val TAG = javaClass<DbHelper>().getSimpleName() val TABLE = "logs" companion object { public val ID: String = "_id" public val TIMESTAMP: String = "TIMESTAMP" public val TEXT: String = "TEXT" } val DATABASE_CREATE = "CREATE TABLE if not exists " + TABLE + " (" + "${ID} integer PRIMARY KEY autoincrement," + "${TIMESTAMP} integer," + "${TEXT} text"+ ")" fun log(text: String) { val values = ContentValues() values.put(TEXT, text) values.put(TIMESTAMP, System.currentTimeMillis()) getWritableDatabase().insert(TABLE, null, values); } fun getLogs() : Cursor { return getReadableDatabase() .query(TABLE, arrayOf(ID, TIMESTAMP, TEXT), null, null, null, null, null); } override fun onCreate(db: SQLiteDatabase) { Log.d(TAG, "Creating: " + DATABASE_CREATE); db.execSQL(DATABASE_CREATE) } override fun onUpgrade(p0: SQLiteDatabase, p1: Int, p2: Int) { } }
由于您使用的是Kotin数据库,使用Anko-Sqlite非常容易。 通过这个库存储和从数据库中检索数据是如此容易。 你可以在这里检查。 https://github.com/Kotlin/anko/wiki/Anko-SQLite#using-anko-sqlite-in-your-project
- 未解决的参考:KOTLIN需要2个版本清理后,拿起代码时使用kapt
- 使用改造方法更具表现力的方式
- 在卡片视图中使回收者视图可点击
- 从Glide库中的缓存中删除图像
- 意图在Android的webview中播放任何嵌入的视频
- Kotlin,Java,multidex,Dagger 2,Butterknife和Realm:transformClassesWithMargingForDebug:重复条目:org / jetbrains / annotations / NotNull.class
- 为什么设置onClickListener一次工作?
- RecyclerView – 正确实施SnackBar UNDO删除(Kotlin)
- FragmentTransaction隐藏/显示有时不起作用