首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >禁用WAL模式

禁用WAL模式
EN

Stack Overflow用户
提问于 2020-01-28 07:54:38
回答 1查看 894关注 0票数 0

我的解决方案似乎行不通,我也不知道为什么。以下是我所尝试的:

逻辑

我的应用程序只有一个数据库实例,如果创建了这个实例,则执行以下操作:

如果当前数据库模式为persist

  • creates,则
  1. 获取当前数据库模式== wal模式,我将其更改为持久化
  2. 更改模式以持久化
  3. ,检查新模式是否为单一数据库实例,并在应用程序的进程存在

G 213时重用它。

代码语言:javascript
运行
复制
// this object holds the ONLY instance of my database and initialises it lazily!
object DBManagerImpl : IDBManager {

    val TAG = "DB-MANAGER"

    internal val database: Database by lazy {

        // 1) change db mode if necessary
        DBJournalUtil.logInfos(null)
        val mode = DBJournalUtil.getDBMode(null)
        L.tag(DBJournalUtil.TAG_WAL_MODE).d { "DB journal_mode: $mode" }
        if (mode.equals(DBJournalUtil.JOURNAL_MODE_WAL, true))
        {
            DBJournalUtil.disableWALMode()
            val mode2 = DBJournalUtil.getDBMode(null)
            L.tag(DBJournalUtil.TAG_WAL_MODE).d { "DB journal_mode - AFTER DISABLE: $mode2" }
        }

        // 2) create single database instance
        L.tag(TAG).d { "database variable created!" }
        val database = Database()

        database
    }
}

一般来说,这是主要代码部分:

代码语言:javascript
运行
复制
    // 1) disable the wal journal mode and change it to persist
    DBJournalUtil.disableWALMode()
    // 2) check the new journal mode => Problem: it's still WAL and not persist
    val mode2 = DBJournalUtil.getDBMode(null)

问题

看起来,我的代码确实正确地更改了数据库模式(基于日志第8行),但是在日志行10中,它再次报告回wal模式。为什么?我错过了什么或者做错了什么?

产生的日志输出

1[数据库日志:14 b]:LOG 2- journal_mode: wal

2[DBManagerImpl$数据库:6e]:DB journal_mode: wal

3[DBJournalUtil:23a]:将DB journal_mode从WAL改为持久化-当前journal_mode = wal

4[DBJournalUtil:35b]:wal_checkpoint -在wal之前(截断):繁忙= 0,LOG = 0,校验点=0

5[DBJournalUtil:6c]:检查点(截断):count =1

6[DBJournalUtil:35b]:wal_checkpoint - AFTER (截断):繁忙= 0,LOG = 0,校验点=0

7[数据库日志:48 a]:setJournalMode: count =1

8[DBJournalUtil:32a]:新journal_mode:持久化// <=这里是持久化!

9[ DBJournalUtil:42 a]:将DB journal_mode从WAL更改为持久化!

10 [ DBManagerImpl$database:12 e]:DB journal_mode - AFTER // <=为什么它仍然是wal模式而不是持久化?

11 [DBManagerImpl$ database :15e]:创建数据库变量!

码- DBJournalUtil

代码语言:javascript
运行
复制
object DBJournalUtil {

    val TAG_WAL_MODE = "DB-WAL-MODE"

    val JOURNAL_MODE_WAL = "WAL"
    val JOURNAL_MODE_PERSIST = "PERSIST"

    val JOURNAL_MODE_DELETE = "DELETE"
    val JOURNAL_MODE_TRUNCATE = "TRUNCATE"

    fun logInfos(db: SQLiteDatabase?) {

        var csr: Cursor? = null
        var dbToUse: SQLiteDatabase? = null

        val dbName = DBManagerProvider.get().dbFileName()
        val dbFile = AppProvider.get().context.getDatabasePath(dbName)

        val logFile = { file: File ->
            val path = file.absolutePath
            val exists = file.exists()
            val size = file.length()
            L.tag(TAG_WAL_MODE).d { "LOG 1 - file: $path | exists: $exists | size: $size" }
        }

        // 1) Dateien loggen
        logFile(dbFile)
        logFile(File(dbFile.getPath() + "-wal")) // WAL Mode file
        logFile(File(dbFile.getPath() + "-shm")) // WAL Mode file

        try {
            dbToUse = db
                    ?: SQLiteDatabase.openDatabase(dbFile.path, null, SQLiteDatabase.OPEN_READWRITE)

            // 2) journal-mode loggen
            csr = dbToUse!!.rawQuery("PRAGMA journal_mode", null)
            if (csr.moveToFirst()) {
                val mode = csr.getString(0)
                L.tag(TAG_WAL_MODE).d { "LOG 2 - journal_mode: $mode" }
            }

        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            csr?.close()
            if (db == null)
                dbToUse?.close()
        }
    }

    fun getDBMode(db: SQLiteDatabase?): String? {

        var mode: String? = null
        var csr: Cursor? = null
        var dbToUse: SQLiteDatabase? = null

        val dbName = DBManagerProvider.get().dbFileName()
        val dbFile = AppProvider.get().context.getDatabasePath(dbName)

        try {
            dbToUse = db
                    ?: SQLiteDatabase.openDatabase(dbFile.path, null, SQLiteDatabase.OPEN_READWRITE)

            csr = dbToUse!!.rawQuery("PRAGMA journal_mode", null)
            if (csr.moveToFirst())
                mode = csr.getString(0)
        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            csr?.close()
            if (db == null)
                dbToUse?.close()
        }

        return mode
    }

    fun disableWALMode() {

        val modeFrom = JOURNAL_MODE_WAL
        val modeTo = JOURNAL_MODE_PERSIST

        val dbName = DBManagerProvider.get().dbFileName()
        val dbFile = AppProvider.get().context.getDatabasePath(dbName)

        var db: SQLiteDatabase? = null

        try {

            // 1) Datenbank öffnen
            db = SQLiteDatabase.openDatabase(dbFile.path, null, SQLiteDatabase.OPEN_READWRITE)

            // 2) Modus auslesen
            val mode = getDBMode(db)
            L.tag(TAG_WAL_MODE).d { "Change DB journal_mode from $modeFrom to $modeTo - current journal_mode = $mode" }

            if (mode == null || mode.equals(modeTo, true)) {
                L.tag(TAG_WAL_MODE).d { "Abbruch da aktueller journal_mode = $mode" }
            } else {

                // 3) wal_checkpoint ausführen
                walCheckpoint(db, "BEFORE wal(TRUNCATE)")

                // 4) wal_checkpoint(TRUNCATE) ausführen um pending operations in DB zu übertragen
                walTruncate(db)

                // 5) wal_checkpoint ausführen
                walCheckpoint(db, "AFTER wal(TRUNCATE)")

                // 6) journal_mode ändern
                setJournalMode(db, modeTo)

                // 7) nochmals aktuellen journal_mode holen
                val newMode = getDBMode(db)
                L.tag(TAG_WAL_MODE).d { "New journal_mode: $newMode" }
            }
        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            db?.close()
        }

        L.tag(TAG_WAL_MODE).d { "Changed DB journal_mode from $modeFrom to $modeTo!" }
    }

    private fun walCheckpoint(db: SQLiteDatabase, logTag: String) {
        var wal_busy = -99
        var wal_log = -99
        var wal_checkpointed = -99

        var csr: Cursor? = null

        try {
            csr = db.rawQuery("PRAGMA wal_checkpoint", null)
            if (csr.moveToFirst()) {
                wal_busy = csr.getInt(0)
                wal_log = csr.getInt(1)
                wal_checkpointed = csr.getInt(2)
            }
        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            csr?.close()
        }

        L.tag(TAG_WAL_MODE).d { "wal_checkpoint - $logTag: Busy = $wal_busy, LOG = $wal_log, CHECKPOINTED = $wal_checkpointed" }
    }

    private fun walTruncate(db: SQLiteDatabase) {
        var csr: Cursor? = null
        try {
            // Checkpoint - Pending Operationen in DB überführen
            csr = db.rawQuery("PRAGMA wal_checkpoint(TRUNCATE)", null)
            val count = csr.count
            L.tag(TAG_WAL_MODE).d { "Checkpoint (TRUNCATE): count = $count" }
        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            csr?.close()
        }
    }

    private fun setJournalMode(db: SQLiteDatabase, mode: String) {
        var csr: Cursor? = null
        try {
            // Checkpoint - Pending Operationen in DB überführen
            csr = db.rawQuery("PRAGMA journal_mode=$mode", null)
            val count = csr.count
            L.tag(TAG_WAL_MODE).d { "setJournalMode: count = $count" }
        } catch (e: Exception) {
            L.tag(TAG_WAL_MODE).e(e)
        } finally {
            csr?.close()
        }
    }
}
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2020-01-28 09:24:14

如果您查看https://sqlite.org/wal.html的第3.3节,您只能保持打开WAL模式,而不能持续关闭WAL模式。您只能将WAL模式作为每个连接的一部分关闭,当您关闭数据库时,该设置将被遗忘。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/59944294

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档