`
javachs
  • 浏览: 123617 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

sqlite的事务和锁

阅读更多
事务
事务定义了一组SQL命令的边界,这组命令或者作为一个整体被全部执行,或者都不执行。事务的典型实例是转帐。
事务的范围
事务由3个命令控制:BEGIN、COMMIT和ROLLBACK。BEGIN开始一个事务,之后的所有操作都可以取消。COMMIT使BEGIN后的所有命令得到确认;而ROLLBACK还原BEGIN之后的所有操作。如:
sqlite> BEGIN;
sqlite> DELETE FROM foods;
sqlite> ROLLBACK;
sqlite> SELECT COUNT(*) FROM foods;
COUNT(*)
412
上面开始了一个事务,先删除了foods表的所有行,但是又用ROLLBACK进行了回卷。再执行SELECT时发现表中没发生任何改变。
SQLite默认情况下,每条SQL语句自成事务(自动提交模式)。
冲突解决
如前所述,违反约束会导致事务的非法结束。大多数数据库(管理系统)都是简单地将前面所做的修改全部取消。
SQLite有其独特的方法来处理约束违反(或说从约束违反中恢复),被称为冲突解决。
如:
sqlite> UPDATE foods SET id=800-id;
SQL error: PRIMARY KEY must be unique
SQLite提供5种冲突解决方案:REPLACE、IGNORE、FAIL、ABORT和ROLLBACK。
REPLACE: 当发违反了唯一完整性,SQLite将造成这种违反的记录删除,替代以新插入或修改的新记录,SQL继续执行,不报错。
IGNORE
FAIL
ABORT
ROLLBACK
数据库锁
在SQLite中,锁和事务是紧密联系的。为了有效地使用事务,需要了解一些关于如何加锁的知识。
SQLite采用粗放型的锁。当一个连接要写数据库,所有其它的连接被锁住,直到写连接结束了它的事务。SQLite有一个加锁表,来帮助不同的写数据库都能够在最后一刻再加锁,以保证最大的并发性。
SQLite使用锁逐步上升机制,为了写数据库,连接需要逐级地获得排它锁。SQLite有5个不同的锁状态:未加锁(UNLOCKED)、共享(SHARED)、保留(RESERVED)、未决(PENDING)和排它(EXCLUSIVE)。每个数据库连接在同一时刻只能处于其中一个状态。每种状态(未加锁状态除外)都有一种锁与之对应。
最初的状态是未加锁状态,在此状态下,连接还没有存取数据库。当连接到了一个数据库,甚至已经用BEGIN开始了一个事务时,连接都还处于未加锁状态。
未加锁状态的下一个状态是共享状态。为了能够从数据库中读(不写)数据,连接必须首先进入共享状态,也就是说首先要获得一个共享锁。多个连接可以同时获得并保持共享锁,也就是说多个连接可以同时从同一个数据库中读数据。但哪怕只有一个共享锁还没有释放,也不允许任何连接写数据库。
如果一个连接想要写数据库,它必须首先获得一个保留锁。一个数据库上同时只能有一个保留锁。保留锁可以与共享锁共存,保留锁是写数据库的第1阶段。保留锁即不阻止其它拥有共享锁的连接继续读数据库,也不阻止其它连接获得新的共享锁。
一旦一个连接获得了保留锁,它就可以开始处理数据库修改操作了,尽管这些修改只能在缓冲区中进行,而不是实际地写到磁盘。对读出内容所做的修改保存在内存缓冲区中。
当连接想要提交修改(或事务)时,需要将保留锁提升为排它锁。为了得到排它锁,还必须首先将保留锁提升为未决锁。获得未决锁之后,其它连接就不能再获得新的共享锁了,但已经拥有共享锁的连接仍然可以继续正常读数据库。此时,拥有未决锁的连接等待其它拥有共享锁的连接完成工作并释放其共享锁。
一旦所有其它共享锁都被释放,拥有未决锁的连接就可以将其锁提升至排它锁,此时就可以自由地对数据库进行修改了。所有以前对缓冲区所做的修改都会被写到数据库文件。
死锁
为什么需要了解锁的机制呢?为了避免死锁。
考虑下面表4-7所假设的情况。两个连接——A和B——同时但完全独立地工作于同一个数据库。A执行第1条命令,B执行第2、3条,等等。
表4-7 一个死锁的假设情况

A连接                                     B连接
sqlite> BEGIN;
                                    sqlite> BEGIN;
                                    sqlite> INSERT INTO foo VALUES('x');
sqlite> SELECT * FROM foo;
                                    sqlite> COMMIT;
                                    SQL error: database is locked
sqlite> INSERT INTO foo VALUES ('x');
SQL error: database is locked

两个连接都在死锁中结束。B首先尝试写数据库,也就拥有了一个未决锁。A再试图写,但当其INSERT语句试图将共享锁提升为保留锁时失败。
为了讨论的方便,假设连接A和B都一直等待数据库可写。那么此时,其它的连接甚至都不能够再读数据库了,因为B拥有未决锁(它能阻止其它连接获得共享锁)。那么时此,不仅A和B不能工作,其它所有进程都不能再操作此数据库了。
如果避免此情况呢?当然不能让A和B通过谈判解决,因为它们甚至不知道彼此的存在。答案是采用正确的事务类型来完成工作。
事务的种类
SQLite有三种不同的事务,使用不同的锁状态。事务可以开始于:DEFERRED、MMEDIATE或EXCLUSIVE。事务类型在BEGIN命令中指定:
BEGIN [ DEFERRED | IMMEDIATE | EXCLUSIVE ] TRANSACTION;
一个DEFERRED事务不获取任何锁(直到它需要锁的时候),BEGIN语句本身也不会做什么事情——它开始于UNLOCK状态。默认情况下就是这样的,如果仅仅用BEGIN开始一个事务,那么事务就是DEFERRED的,同时它不会获取任何锁;当对数据库进行第一次读操作时,它会获取SHARED锁;同样,当进行第一次写操作时,它会获取RESERVED锁。
由BEGIN开始的IMMEDIATE事务会尝试获取RESERVED锁。如果成功,BEGIN IMMEDIATE保证没有别的连接可以写数据库。但是,别的连接可以对数据库进行读操作;但是,RESERVED锁会阻止其它连接的BEGIN IMMEDIATE或者BEGIN EXCLUSIVE命令,当其它连接执行上述命令时,会返回SQLITE_BUSY错误。这时你就可以对数据库进行修改操作了,但是你还不能提交,当你COMMIT时,会返回SQLITE_BUSY错误,这意味着还有其它的读事务没有完成,得等它们执行完后才能提交事务。
EXCLUSIVE事务会试着获取对数据库的EXCLUSIVE锁。这与IMMEDIATE类似,但是一旦成功,EXCLUSIVE事务保证没有其它的连接,所以就可对数据库进行读写操作了。
上节那个例子的问题在于两个连接最终都想写数据库,但是它们都没有放弃各自原来的锁,最终,SHARED锁导致了问题的出现。如果两个连接都以BEGIN IMMEDIATE开始事务,那么死锁就不会发生。在这种情况下,在同一时刻只能有一个连接进入BEGIN IMMEDIATE,其它的连接就得等待。BEGIN IMMEDIATE和BEGIN EXCLUSIVE通常被写事务使用。就像同步机制一样,它防止了死锁的产生。
基本的准则是:如果你正在使用的数据库没有其它的连接,用BEGIN就足够了。但是,如果你使用的数据库有其它的连接也会对数据库进行写操作,就得使用BEGIN IMMEDIATE或BEGIN EXCLUSIVE开始你的事务。
分享到:
评论

相关推荐

    SQLite教程(十二):锁和并发控制详解

    在SQLite中,锁和并发控制是数据库管理系统中关键的组成部分,它们确保了数据的一致性和完整性。SQLite使用了一种称为pager_module的模块来处理这些任务,该模块遵循ACID原则,即原子性、一致性、隔离性和持久性。在...

    C语言SQLite3事务和锁的操作实例

    在介绍C语言SQLite3事务和锁的操作实例之前,我们先来了解SQLite3的基础知识以及C语言如何与SQLite3进行交互。 SQLite是一个轻量级的数据库系统,它将整个数据库存储在一个单一的文件中。作为一种嵌入式数据库,它...

    sqlite死锁datebaselock解决方案

    在多线程环境下,由于并发操作不当,可能会出现“database is locked”(数据库被锁定)的错误,这通常涉及到SQLite的锁机制和事务处理。本文将深入探讨这个问题,并提供具体的解决方案。 一、SQLite锁机制 SQLite...

    SQLite3数据库中的文件锁和同步机制

    PagerModue负责SQLite事务的ACID,也提供缓存功能。PagerModue不需要知道BTree,字符编码, 索引的结构, Pager Module用来管理Page, 一个Page对应一个DiskBlock, 大小一般是1024Byte。  1. SQLite3 数据库的锁...

    SQLITE数据库 UPDATE慢

    4. **事务处理**:SQLite默认使用自动提交模式,每次操作都会开启和结束一个事务。大量的UPDATE操作会导致频繁的磁盘I/O,影响性能。将多个UPDATE语句放在同一个BEGIN...COMMIT事务中,可以减少磁盘I/O次数,提高...

    sqlite源码

    7. **锁机制**:SQLite使用共享锁和独占锁来处理并发访问。了解锁的获取和释放策略有助于优化多线程环境下的性能。 8. **错误处理和日志记录**:SQLite源码中包含了丰富的错误检测和处理机制,以及详细的日志记录,...

    sqlite3 sqlite3 sqlite3

    SQLite3 是一个开源的、自包含的、无服务器的、零配置的、事务性的 SQL 数据库引擎。它被广泛应用于嵌入式系统和移动设备,因为它具有轻量级、高效且可靠的特性。SQLite3 提供了 SQL 接口,允许用户通过编程语言与...

    解决sqlite死锁示例异常database is locked示例

    总结来说,SQLite的死锁问题主要源于资源竞争和事务顺序,通过理解死锁产生的原因,结合异常处理、事务设计优化、日志监控等手段,可以有效地预防和解决"database is locked"异常。在实际开发中,应当重视数据库的...

    sqlite源码及分析

    5. 锁机制:SQLite采用了页级锁定策略,根据工作模式的不同,可以实现共享锁和独占锁,以协调多个并发连接对数据库的访问。 入门教程: 对于初学者,可以从以下几个方面入手学习SQLite: 1. SQL基础:熟悉SQL语言...

    C#多线程读写sqlite

    这个项目的源代码可能包含了如何在C#中使用多线程和同步锁与SQLite交互的示例。通常,它会展示如何在多个线程中并行执行读写操作,以及如何使用`lock`或其他同步原语来保护对数据库的访问。测试部分可能会比较不同...

    QT中sqlite多线程操作4个注意问题

    然而,当在多线程环境中使用SQLite时,需要注意一些关键问题以确保数据的安全性和一致性。以下是四个重要的考虑因素: 1. **线程安全**: SQLite本身并不提供完全的线程安全,这意味着在不同线程中并发访问数据库...

    SQLite权威指南 chm

    6. **SQLite内核**:这部分深入到SQLite的内部工作机制,包括页缓存、事务机制、锁管理、B树结构等。通过对SQLite内核的了解,读者可以更好地理解和解决实际开发中遇到的问题,同时也能为优化数据库性能提供理论依据...

    VB6.0 操作SQLite 数据库的完整示例代码

    SQLite是一种自包含、无服务器、零配置、事务性的SQL数据库引擎,广泛应用于移动设备、嵌入式系统以及桌面应用程序。本示例代码提供了全面的功能,使开发者能够轻松地在VB6.0项目中执行数据库的各种操作。 首先,...

    Sqlite3源代码

    通过阅读和理解SQLite3的源代码,开发者可以学习到数据库设计、查询优化、事务处理、锁机制等关键概念,这对于深入理解数据库系统和开发自己的数据库软件大有裨益。同时,SQLite3源代码的简洁性和高效性也使其成为...

    svn清理sqlite3

    2. **数据库日志清理**:SQLite3在处理事务时会产生wal(Write-Ahead Logging)日志文件和shm文件,如`.svn/wc.db-wal`和`.svn/wc.db-shm`。如果这些文件未正常清理,可能导致svn操作异常。你可以手动删除这两个文件...

    Android SQLite源码+说明

    它可能包含以下内容:SQLite数据库的创建和打开,如何使用SQLiteOpenHelper类来管理数据库版本,SQLite表的创建语句,如何执行SQL查询,事务处理,以及如何在Android应用中插入、更新和删除数据。文档可能还会涉及...

    sqlite数据库c++类封装,及源码实例,boost多线程

    为了确保线程安全,我们需要考虑锁机制,如互斥锁(`mutex`)和条件变量(`condition_variable`),以防止多个线程同时访问SQLite数据库导致的数据竞争问题。 在实际应用中,可能会有一个`Database`类,它包含了...

Global site tag (gtag.js) - Google Analytics