SQLiteHelper类||java.lang.IllegalStateException: attempt to re-open an already-closed object

来源:互联网 发布:天天向上的网络作家 编辑:程序博客网 时间:2024/06/14 21:19

在实际的android开发项目中很少使用SQLiteDatabase的方法来打开数据库,通常都会继承SQLiteOpenHelper开发子类,并通过该子类的

getReadableDatabase(),getWritableDatabase()方法打开数据库。

类概述:


一个android提供的管理数据库的工具类,帮助创建数据库和数据库版本管理。


一般的使用使创建一个SQLiteOPenHelper子类来扩展其onCreate(SQLiteDatabase), onUpgrade(SQLiteDatabase, int, int)方法,同时任意实现onOpen(SQLiteDatabase)方法,同时打开数据库操作必须保证数据库存在,如果不存在则创建它,并且对其必要的升级,维护其保持一个最佳的状态。


使用本类提供内容开始创建数据库是非常容易的,首先必须对数据库进行升级,以避免在数据库启动后长期使用而阻塞数据。


SQLiteOpenHelper包含如下方法:


synchronized SQLiteDatabase getReadableDatabse():以读写的方式打开数据库对应的SQLiteDatabase对象


synchronized SQLiteDatabase getWritableDatabse():以写的方式打开数据库对应的SQLiteDatabase对象


sbstract void onCreate(SQLiteDatabase db):当第一次创建数据库的时候回调该方法。


当调用SQLiteOpenHelper的getReadableDatabase()或getWritableDatabase()方法获得取用与操作数据库的SQLiteDatabase实例时,如果数据库不存在,android系统会自动生成一个数据库,接着调用onCreate()方法,onCreate()方法在初次生成数据库时才会被调用,重写onCreate()方法时,可以生成数据库表结构及添加一些应用使用到的初始化的数据。


sbstract void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion):当数据库版本更新时回调该方法。


该方法调用时oldVersion代表数据库之前的版本号,newVersion代表当前的数据库的版本号。

--------------------------------------------那么在哪里指定数据库的版本号呢?当程序创建SQLiteOpenHelper对象时,必须指定一个veriosn参数,该参数就决定了所使用的数据库的版本,也就是说,数据库版本是有程序员控制的。只要某次创建SQLiteOpenHelper时指定的数据库版本高于之前指定的版本号,系统就会自动触发onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)方法,程序就可以在onUpgrade()方法里面根据原版本号和目标版本号进行判断,即可根据版本号进行必须的表结构更新。


synchronized void close():关闭所有打开的SQLiteDatabase。


**************************************************************************************************************************************************************************

类介绍完了说说java.lang.IllegalStateException: attempt to re-open an already-closed object这个错误:


出现原因:数据库插入时不能判断是否重复(主键ID是自动生成的),所以需要在插入的时候进行查找。。。。


我在一个数据库插入方法中调用了另一个数据库查询方法,我的数据库查询方法都是在开始的时候获取SQLiteDatabase对象,在结束的时候关闭SQLiteDabse对象,结果内部的数据库查询方法在结束的时候直接关闭了SQLiteDatabase对象,导致外面的数据库查询操作报错,在这里大家不要以为多获取了几个SQLiteDatabase对象就可以了,每个线程只能使用一个SQLiteOpenHelper,也就使得每个线程使用一个SQLiteDatabase对象(多线程操作数据库会报错);


解决办法就是我不再关闭内部数据库查询方法的SQLiteDatbase对象或者将那个方法直接集成到外面的查询方法中,当然,要确保这个查询方法只会出现其他数据库查询方法中,要是单独用这个方法,反而会因为SQLiteDatabase对象没有关闭而报错;


0 0
原创粉丝点击