Android SQLite数据库:SQLiteDiskIOException:磁盘I/O错误(代码1802)

9

我有一个使用SQLite数据库的应用程序。

当应用程序启动时,我调用RSSDatabase的构造函数创建数据库,这个过程运行良好。点击按钮后,我使用AsyncTask从RSS源获取数据,并将数据插入到数据库中,但是在调用db.insert("posts", "", values)时会出现错误。以下是logcat的错误信息:

02-04 21:43:22.050  19154-19357/com.example.aolreader E/SQLiteLog﹕ (1802) os_unix.c:30026: (2) stat(/data/data/com.example.aolreader/databases/rss_db) -
02-04 21:43:22.050  19154-19357/com.example.aolreader E/SQLiteLog﹕ (1802) statement aborts at 13: [INSERT INTO posts(author,title,date,description,link) VALUES (?,?,?,?,?)]
02-04 21:43:22.050  19154-19357/com.example.aolreader E/SQLiteDatabase﹕ Error inserting author=AP title=UK admits ‘limited’ role in Operation Blue Star  date=Tue, 4 Feb 2014 19:03:27 +0530 description=
    The British government had recently ordered an urgent investigation into possible U.K. involvement in the raid.
    link=http://www.thehindu.com/news/international/world/uk-admits-limited-role-in-operation-blue-star/article5653316.ece
    android.database.sqlite.SQLiteDiskIOException: disk I/O error (code 1802)
            at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Native Method)
            at android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:782)
            at android.database.sqlite.SQLiteSession.executeForLastInsertedRowId(SQLiteSession.java:788)
            at android.database.sqlite.SQLiteStatement.executeInsert(SQLiteStatement.java:86)
            at android.database.sqlite.SQLiteDatabase.insertWithOnConflict(SQLiteDatabase.java:1469)
            at android.database.sqlite.SQLiteDatabase.insert(SQLiteDatabase.java:1339)
            at com.example.aolreader.RSSDatabase.insertPost(RSSDatabase.java:44)
            at com.example.aolreader.XMLRSSActivity.parseNews(XMLRSSActivity.java:386)
            at com.example.aolreader.XMLRSSActivity.access$000(XMLRSSActivity.java:33)
            at com.example.aolreader.XMLRSSActivity$ParseTask.doInBackground(XMLRSSActivity.java:171)
            at com.example.aolreader.XMLRSSActivity$ParseTask.doInBackground(XMLRSSActivity.java:134)
            at android.os.AsyncTask$2.call(AsyncTask.java:288)
            at java.util.concurrent.FutureTask.run(FutureTask.java:237)
            at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
            at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
            at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
            at java.lang.Thread.run(Thread.java:841)

我在谷歌上搜索了这个错误,并发现它可能是由于不同的线程同时访问数据库所致,但我查看了我的代码,并没有发现多个线程存在竞争条件的情况。
以下是我的数据库帮助类:
class RSSDatabaseHelper extends SQLiteOpenHelper {

    private static final String createPostTableSql = "create table posts (" +
            "_id integer primary key autoincrement," +
            "title varchar," +
            "author varchar," +
            "link varchar," +
            "description varchar," +
            "date varchar)";

    private Context context;

    public RSSDatabaseHelper(Context c, String name, SQLiteDatabase.CursorFactory factory, int version) {
        super(c, name, factory, version);
        context = c;
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL(createPostTableSql);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        db.execSQL("drop table if exists posts");
        onCreate(db);
    }
}

我的数据库类:

public class RSSDatabase {
    private static RSSDatabase instance;

    private SQLiteDatabase db;
    private RSSDatabaseHelper dbHelper;

    private RSSDatabase(Context c) {
        dbHelper = new RSSDatabaseHelper(c, "rss_db", null, 1);
        db = dbHelper.getWritableDatabase();
    }

    public static synchronized RSSDatabase getDatabaseInstance(Context c) {
        if (instance == null) {
            instance = new RSSDatabase(c);
        }
        return instance;
    }

    public void insertPost(Post post) {
        ContentValues values = new ContentValues();
        values.put("title", post.title);
        values.put("author", post.author);
        values.put("link", post.link);
        values.put("description", post.description);
        values.put("date", post.date);
        db.insert("posts", "", values);
    }
}

有谁能解释一下这个问题吗?

提前感谢!


检查内容值中是否存在任何空值。 - Triode
@RajeshCP 我相信那不是问题,因为在我的表定义中没有“NOT NULL”。 - Guanlun
这是本地级别的 SQLITE_IOERR_FSTAT。你的数据库有多大?有并发的读/写/删除操作吗? - laalto
@laalto 数据库几乎为空。我相信数据库没有同时被多个线程访问。 - Guanlun
@laalto 但是由于数据库插入是在AsyncTask中执行的,可能会出现一些并发问题。我正在检查。 - Guanlun
您可以在此处查看 https://stackoverflow.com/questions/44312563/sqlitediskioexception-disk-i-o-error-code-1802-while-compiling-pragma-jou/45329480#45329480 - varotariya vajsi
1个回答

18

当应用程序启动时,调用RSSDatabase的构造函数会锁定UI线程上的数据库。

db = dbHelper.getWritableDatabase(); // In the constructor

现在从异步任务(不同的线程),您正在尝试向由UI线程锁定的数据库中插入数据。这将抛出异常,因为第一次写操作对数据库进行了锁定。


谢谢!在将代码移出UI线程后,这正是我发现的。 - Guanlun
4
等等,为什么它不会抛出“数据库被锁定(代码5)”而是抛出“磁盘I/O错误(代码10)”? - Michael
1
@Guanlun,你是如何解决这个数据库锁问题的?我因为这个问题被阻塞了。 - krishna

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接