SQLite中的隔离(八)

 返回:SQLite—系列文章目录   

上一篇:SQLite版本3中的文件锁定和并发(七)

下一篇:SQLite 查询优化器概述(九)

数据库的“isolation”属性确定何时对 一个操作的数据库对其他并发操作可见。

数据库连接之间的隔离

如果使用两个不同的数据库连接读取和写入同一个数据库(两个不同的 sqlite3 对象返回 单独调用 sqlite3_open()) 和两个数据库连接 没有共享缓存,则读取器只能 请参阅编写器的完整已提交事务。部分更改 作者没有承诺的对读者来说是不可见的。 无论两个数据库连接是否在 同一线程,在同一进程的不同线程中,或在 不同的过程。这 是 SQL 数据库系统的常见行为和预期行为。

上一段也是正确的(单独的数据库连接是 彼此隔离)在共享缓存模式下,只要read_uncommitted编译指示保持关闭状态即可。默认情况下,read_uncommitted编译指示处于关闭状态,因此,如果应用程序不执行任何操作来打开它, 它将保持关闭状态。因此,除非使用read_uncommitted编译指示 若要更改默认行为,请由一个数据库连接所做的更改 对于共享 相同的缓存,直到写入器提交其事务。

如果两个数据库连接共享同一个缓存,并且读取器具有 启用了read_uncommitted编译指示,然后读者将能够 请参阅编写器在编写器事务提交之前所做的更改。 共享缓存模式和read_uncommitted指示的组合使用是一个数据库连接可以查看未提交的更改的唯一方法 在不同的数据库连接上。在所有其他情况下,分开 数据库连接彼此完全隔离。

除了打开PRAGMA read_uncommitted共享缓存数据库连接的情况外,SQLite中的所有事务都显示 “可序列化”隔离。SQLite 实现可序列化事务 通过实际序列化写入。只能有一个作家 一次到一个 SQLite 数据库。可以有多个数据库连接 同时打开,所有这些数据库连接都可以写入 到数据库文件,但他们必须轮流。SQLite使用锁 自动序列化写入;这不是什么 使用SQLite的应用程序需要担心。

隔离和并发

SQLite 使用 与数据库文件位于同一目录中的暂时性日志文件。 有两种主要的“日志模式”。 较旧的“回滚模式”对应于使用“DELETE”、“PERSIST”、 或“TRUNCATE”选项添加到journal_mode编译指示中。在回滚模式下, 更改将直接写入数据库文件,同时 构造一个单独的回滚日志文件,该文件能够恢复 如果事务回滚,则数据库恢复到其原始状态。 回滚模式(特别是 DELETE 模式,即回滚日志 在每次事务结束时从磁盘中删除)是当前 默认行为。

从3.7.0版(2010-07-21)开始, SQLite还支持“WAL模式”。在 WAL 模式下, 更改不会写入原始数据库文件。相反,更改 进入单独的“预写日志”或“WAL”文件。 后来,交易后 提交时,这些更改将从 WAL 文件移回 原始数据库在名为“checkpoint”的操作中。WAL 模式是 通过运行“PRAGMA journal_mode=WAL”启用。

在回滚模式下,SQLite通过锁定数据库来实现隔离 文件并防止其他数据库连接读取 而每个写入事务都在进行中。 读者可以在写作开始时,在任何内容之前处于活动状态 刷新到磁盘,而所有更改仍保存在编写器的 专用内存空间。但在对数据库文件进行任何更改之前 在磁盘上,必须(暂时)驱逐所有读取器才能给写入器 对数据库文件的独占访问。 因此,禁止读者看到不完整的内容 由于被锁定在数据库之外的事务,而 事务正在写入磁盘。只有在交易之后 完全写入并同步到磁盘并提交是允许的读取器 返回到数据库中。因此,读者永远没有机会看到部分内容 书面更改。

WAL 模式允许同时读取和写入。它可以这样做,因为 更改不会覆盖原始数据库文件,而是执行 添加到单独的预写日志文件中。这意味着读者可以继续 从原始数据库文件中读取旧的、原始的、未更改的内容 同时,编写器将追加到预写日志中。 在WAL模式下,SQLite表现出“快照隔离”。当读取事务时 开始,该读者继续看到数据库的不变“快照” 文件,因为它在读取事务开始时存在。 在读取事务 active 对读取事务仍然不可见,因为读取器是 查看前一时刻的数据库文件快照。

举个例子:假设有两个数据库连接 X 和 Y。 使用 BEGIN 后跟一个或多个 SELECT 语句的读取事务。 然后 Y 出现并运行 UPDATE 语句来修改数据库。 X 随后可以对 Y 修改的记录执行 SELECT,但 X 将看到较旧的未修改条目,因为 Y 的更改都是 当 X 持有读取事务时,X 不可见。如果 X 想看 Y 所做的更改,则 X 必须结束其读取事务,并且 启动一个新的(通过运行 COMMIT,然后运行另一个 BEGIN)。

另一个示例:X 使用 BEGIN 和 SELECT 启动读取事务,然后 Y 使用 UPDATE 对数据库进行更改。然后 X 尝试做一个 使用 UPDATE 更改为数据库。X 试图升级其 从读取事务到写入事务的事务失败并显示SQLITE_BUSY_SNAPSHOT错误,因为数据库的快照正在 查看的 X 不再是数据库的最新版本。如果 X 是 允许写入,它将分叉数据库文件的历史记录,即 SQLite不支持的东西。为了让 X 写入数据库, 它必须首先发布其快照(例如使用 ROLLBACK),然后 使用后续 BEGIN 启动新事务。

如果 X 启动一个事务,该事务最初只会读取但 X 知道它 最终会想写,不想被烦恼 由于另一个连接而可能出现的SQLITE_BUSY_SNAPSHOT错误 在队列中跳到它前面,然后 X 可以发出 BEGIN IMMEDIATELY 来启动 它的交易而不仅仅是普通的 BEGIN。BEGIN IMMEDIATE 命令继续并启动写入事务,从而阻止所有 其他作家。如果 BEGIN IMMEDIATE 操作成功,则没有 该事务中的后续操作将失败并出现SQLITE_BUSY错误。

同一数据库连接上的操作之间没有隔离

SQLite 在单独的数据库中提供操作之间的隔离 连接。但是,操作之间没有隔离 发生在同一数据库连接中。

换言之,如果 X 使用 BEGIN IMMEDIATE 开始写入事务,然后发出一个或多个 UPDATE、DELETE 和/或 INSERT 语句,则这些更改对后续 SELECT 语句可见 在数据库连接 X 中计算的。 不同的数据库连接 Y 将不显示任何更改,直到 X 事务提交。但是 X 中的 SELECT 语句将显示更改 在提交之前。

在单个数据库连接 X 中,SELECT 语句始终看到所有 在 SELECT 开始之前完成的对数据库的更改 语句,无论是已提交的还是未提交的。和 SELECT 语句 显然看不到 SELECT 语句之后发生的任何更改 完成。但是,在 SELECT 语句中发生的更改呢? 正在跑步吗?如果启动了 SELECT 语句,并且 sqlite3_step() 接口单步执行了大约一半的输出,则某些 UPDATE 语句由修改表的应用程序运行,该 SELECT 语句正在读取,然后对 sqlite3_step() 进行更多调用 完成 SELECT 语句?将 SELECT 的后续步骤 声明是否看到 UPDATE 所做的更改?答案是 此行为未定义。特别是,无论 SELECT 语句是否 看到并发更改取决于 SQLite 的哪个版本 running,数据库文件的架构,无论 ANALYZE 是否具有 已运行,以及查询的详细信息。在某些情况下,这可能取决于 在数据库文件的内容上也是如此。没有好的方法可以知道是否 或者 SELECT 语句不会看到对数据库所做的更改 通过启动 SELECT 语句后的同一数据库连接。 因此,开发人员应该努力避免编写应用程序 对在这种情况下会发生什么做出假设。

如果应用程序在单个表上发出 SELECT 语句,例如 “SELECT rowid, * FROM table WHERE ...” 并开始单步执行 使用 sqlite3_step() 的该语句的输出并检查每个 行,则应用程序可以安全地删除当前行或 使用“DELETE FROM table WHERE rowid=?”的任何前行。它也是安全的 (从某种意义上说,它不会损害数据库)让应用程序 删除预计稍后在查询中出现但尚未出现的行 出现了。但是,如果删除了将来的行,则可能会发生以下情况 该行在随后的 sqlite3_step() 之后出现,即使在它有 据称已被删除。或者它可能不是。该行为是未定义的。 应用程序可以 此外,当 SELECT 语句 正在运行,但是否显示新行 在查询的后续 sqlite3_step() 中未定义。和应用程序 可以更新当前行或任何前一行,但这样做可能会导致 该行将在随后的 sqlite3_step() 中重新出现。只要 应用程序准备处理这些歧义,操作 它们本身是安全的,不会损害数据库文件。

就前两段而言,两个数据库连接 具有相同共享缓存且已启用 PRAGMA read_uncommitted的数据库连接被视为相同的数据库连接。

总结

  1. SQLite 中的事务是可序列化的。

  2. 在一个数据库连接中所做的更改对所有其他数据库都是不可见的 提交前的连接。

  3. 查询将查看在同一数据库连接上完成的所有更改 在查询开始之前,无论这些更改是否发生 已经提交。

  4. 如果查询后同一数据库连接发生更改 开始运行,但在查询完成之前,则不确定是否 或者查询不会看到这些更改。

  5. 如果查询后同一数据库连接发生更改 开始运行,但在查询完成之前,查询可能会返回 多次更改的行,或者它可能会返回以前更改的行 删除。

  6. 就前四项而言,两个数据库连接 使用相同的共享缓存,并且启用 PRAGMA read_uncommitted 被视为相同的数据库连接,而不是单独的数据库 连接。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/295746.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Hadoop安装部署-SecondaryNameNode备份版

Hadoop分布式文件系统支持NameNode节点的高可用性,本文主要描述Secondary NameNode数据备份版本的安装部署。 如上所示,NameNode主节点同步数据到NameNode备份节点,当NameNode主节点发生故障变得不可用时, NameNode主节点重新启动…

关联对象介绍

关联对象的作用 在分类里面,不可以直接为分类添加属性 在代理中,不可以直接为代理添加属性 在普通类中,property (assign, nonatomic) int age; 会做三件事: 生成age的成员变量生成age的get、set方法的声明生成age的get、set方…

【论文通读】UFO:A UI-Focused Agent for Windows OS Interaction

UFO:A UI-Focused Agent for Windows OS Interaction 前言AbstractMotivationMethodsExperimentConclusion 前言 Windows客户端第一个JARVIS,利用GPT4 Vision识别截图信息辅助智能体自动化执行操作,作为微软大肆宣传的一篇工作,其…

如何使用 Python 本地客户端操作读写云服务器 Redis 缓存数据库详细教程(更新中)

Redis 基本概述 Redis(Remote Dictionary Server)是一个开源的使用 ANSI C 语言编写的、支持网络、可基于内存亦可持久化的日志型、Key-Value 数据库,并提供多种语言的 API。它通常被称为数据结构服务器,因为值(value…

【OpenCV】 基础入门(一)初识 Mat 类 | 通过 Mat 类显示图像

🚀 个人简介:CSDN「博客新星」TOP 10 , C/C 领域新星创作者💟 作 者:锡兰_CC ❣️📝 专 栏:【OpenCV • c】计算机视觉🌈 若有帮助,还请关注➕点赞➕收藏&#xff…

部署项目遇到的各种问题总结

文章目录 前言一、后端问题 jar包运行出现错误宝塔面板使用jdk17二、数据库问题 版本问题三、前端问题 连不上后端总结 前言 在做完项目之后,为了让别人访问到自己的网站,就需要部署前端后端以及数据库,但是在部署的过程中出现了各种问题和困…

docker 部署 nali 开源 IP 地理信息归属查询软件

前言 早前用到一个小巧开源的 IP 归属地查询软件,官方提供了 Dockerfile,使用了一段时间觉得还不错,非常简单便捷。 部署 docker 启动 由于该项目会在首次启动自动下载 IP 数据库,所以最好通过挂载目录的方式,将数据库目录挂在到本地,避免…

【C语言】2048小游戏【附源码】

欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 一、游戏描述: 2048是一款数字益智类游戏,玩家需要使用键盘控制数字方块的移动,合并相同数字的方块,最终达到数字方块上出现“2048”的目标。 每次移动操作,所…

C++进阶--C++11(1)

C11是C编程语言的一个版本,于2011年发布。C11引入了许多新特性,为C语言提供了更强大和更现代化的编程能力。这篇文章将对C11的一些新增特性进行讲解和实际应用场景。 统一的列表初始化 {}初始化 在C98中,使用{}符号的一般只仅限于对数组和…

基于蚁群算法的三维路径规划(matlab实现)

作品简介 1 理论基础 1.1 三维路径规划问题概述 三维路径规划指在已知三维地图中,规划出一条从出发点到目标点满足某项指标最优,并且避开了所有三维障碍物的三维最优路径。现有的路径规划算法中,大部分算法是在二维规划平面或准二维规划平面…

今日头条signature参数js逆向(爬虫)

今日头条是ajax动态加载 话不多说,直接上代码 windowglobal;window.location{"ancestorOrigins": {},"href": "https://www.toutiao.com/","origin": "https://www.toutiao.com","protocol": "…

连接Redis不支持集群错误,ERR This instance has cluster support disabled,解决方案

1. 问题背景 调整redis的配置后,启动程序时, 会报如下错误: [redis://172.16.0.8xxx]: ERR This instance has cluster support disabledSuppressed: io.lettuce.core.RedisCommandExecutionException: ERR This instance has cluster supp…

【C++】二分查找算法(模板)

重点 只需要记住两点: 1.left right 时,一定就是最终结果(包括找不到目标值),无需再次判断,如果判断就会死循环 2.求中点如果是求左端点 mid left (right - left)/2 如果是求右端点 mid left (right -…

MATLAB 自定义均值滤波 (53)

MATLAB 自定义均值滤波 (53) 一、算法介绍二、算法实现1.原理2.代码一、算法介绍 均值滤波,是一种常见的点云平滑算法,改善原始点云的数据质量问题,MATLAB自带的工具似乎不太友好,这里提供自定义实现的点云均值滤波算法,具体效果如下所示: 均值滤波前: 均值滤波后:…

计算机网络 - 基础篇总结

TCP/IP 网络模型有哪几层? 1.应用层 为用户提供应用功能 2.传输层 负责为应用层提供网络支持 使用TCP和UDP 当传输层的数据包大小超过 MSS(TCP 最大报文段长度) ,就要将数据包分块,这样即使中途有一个分块丢失或损坏…

算法学习——LeetCode力扣图论篇2(1020. 飞地的数量、130. 被围绕的区域、827. 最大人工岛)

算法学习——LeetCode力扣图论篇2 1020. 飞地的数量 1020. 飞地的数量 - 力扣(LeetCode) 描述 给你一个大小为 m x n 的二进制矩阵 grid ,其中 0 表示一个海洋单元格、1 表示一个陆地单元格。 一次 移动 是指从一个陆地单元格走到另一个相…

【linux】lsof命令使用

1. 功能 lsof list open files, 列出被进程所使用的文件名称。 2. 基础语法 3. 参数含义 参数含义-a过滤出多个选项要同时满足的文件-U仅列出UNIX-like系统的socket文件类型。-u指定用户,比如-u atiaisi,会把用户atiaisi相关的进程使用的文件列出来。…

游戏运营分析:如何在新游戏上线初期实现精细化运营?

一、背景介绍 在当今的手游市场中,每一款新游戏的发布都如同踏上一段充满未知与挑战的探险之旅。游戏刚上线时,运营情况往往如同飘摇的小船,随时可能受到风浪的侵袭。此时,如何准确地找到问题所在,为游戏的健康运营和持…

SAD法(附python实现)和Siamese神经网络计算图像的视差图

1 视差图 视差图:以左视图视差图为例,在像素位置p的视差值等于该像素在右图上的匹配点的列坐标减去其在左图上的列坐标 视差图和深度图: z f b d z \frac{fb}{d} zdfb​ 其中 d d d 是视差, f f f 是焦距, b b…

openGauss学习笔记-254 openGauss性能调优-使用Plan Hint进行调优-子链接块名的hint

文章目录 openGauss学习笔记-254 openGauss性能调优-使用Plan Hint进行调优-子链接块名的hint254.1 功能描述254.2 语法格式254.3 参数说明254.4 示例 openGauss学习笔记-254 openGauss性能调优-使用Plan Hint进行调优-子链接块名的hint 254.1 功能描述 指明子链接块的名称。…