关系数据库的事务隔离、锁定与并发控制

2024-04-29 18:18

本文主要是介绍关系数据库的事务隔离、锁定与并发控制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

事务隔离(Isolation),指的是在数据库系统中并发事务(Transaction)之间的可见性,以及如何相互影响的定义。事务隔离是ACID的四个特性(原子性、一致性、隔离性、持久性)之一。在ANSI/ISO的SQL标准中,定义了四个事务隔离级别,分别用于不同的场合。在传统的数据库系统的实现中,一般实现不同等级的隔离性的方法是使用锁定(lock),根据具体需求可细分为读取锁(read lock)写入锁(write lock)范围锁(range-locks)

可序列化

可序列化(SERIALIZABLE)最高的隔离级别。在此级别下,所有事务的完整性都被保留,这意味着所有的事务可以被序列化地执行。当只有两个事务之前没有任何冲突时,才能并发地执行。

可重复读取

可重复读取(REPEATABLE READS)级别下,数据库系统会在在整个事务期间保持所有读取锁写入锁,但相较于可序列化范围锁不会被管理,所以幻象读取(phantom reads)可能会出现。

授权读取

授权读取(READ COMMITTED)级别下,数据库系统在整个事务期间保持写入锁,但读取锁会在SELECT执行后立即释放,所以不可重复读取(non-repeatable reads)可能会出现。

未授权读取

未授权读取(READ UNCOMMITTED)最低的隔离级别。这个级别允许出现肮脏读取(dirty reads)

下列的示例解释了幻象读取(phantom reads)不可重复读取(non-repeatable reads)肮脏读取(dirty reads)。数据在下表中定义:

users
idnameage
1Joe20
2Jill25

肮脏读取

当一个事务试图读取另一个还未提交的事务正在修改的某一行数据时,肮脏读取(dirty reads)就会发生。

下列示例中,事务2正在修改某行,但还没有提交。事务1试图读取这一行。如果事务2回滚了(rolls back)变更,或者后面又进行了其他的修改,那么事务1就获得了肮脏(dirty)的数据。

事务1事务2
sql/* Query 1 */SELECT * FROM users WHERE id = 1;


sql/* Query 2 */UPDATE users SET age = 21 WHERE id = 1;/* No commit here */
sql/* Query 1 */SELECT * FROM users WHERE id = 1;


sqlROLLBACK; /* lock-based DIRTY READ */

不可重复读取

当一个事务正在执行的时候,对某一行两次读取的结果不一致,则称发生了不可重复读取(non-repeatable reads)

以下示例中事务1读取了某行,之后事务2立刻修改了这一行并提交了结果,事务1再读取这一行的时候,结果就不一致了。

事务1事务2
sql/* Query 1 */SELECT * FROM users WHERE id = 1;


sql/* Query 2 */UPDATE users SET age = 21 WHERE id = 1;COMMIT; /* in multiversion concurrency control, or lock-based READ COMMITTED */
sql/* Query 1 */SELECT * FROM users WHERE id = 1;COMMIT; /* lock-based REPEATABLE READ */

幻象读取

幻象读取(phantom reads)指的是两次集合查询之间返回了不一致的结果。以下示例展现了这一现象。

事务1事务2
sql/* Query 1 */SELECT * FROM usersWHERE age BETWEEN 10 AND 30;


sql/* Query 2 */INSERT INTO users VALUES ( 3, 'Bob', 27 );COMMIT;
sql/* Query 1 */SELECT * FROM usersWHERE age BETWEEN 10 AND 30;

隔离级别与读取现象

隔离级别肮脏读取不可重复读取幻象读取
未授权读取可能发生可能发生可能发生
授权读取-可能发生可能发生
可重复读取--可能发生
可序列化---

隔离级别与锁定

隔离级别写入锁读取锁范围锁
未授权读取---
授权读取需要--
可重复读取需要需要-
可序列化需要需要需要

乐观锁与多版本并发控制

在锁定控制的数据库系统中,死锁(dead lock)指的是两个以上的事务互相依赖等待,从而都被阻塞的现象。与多线程程序的设计不同,锁定控制的数据库中死锁出现是很正常现象,而且是无法根本上避免的。当死锁出现并且被数据库系统检测到时,所有死锁的事务都会被驳回,用户不得不根据需要进行重提交。然而死锁不仅检测代价是很高昂的,而且还会浪费大量资源,如果死锁频繁出现,会大大降低数据库系统的并发性能。但我们不能为了避免死锁而降低隔离等级,而且有一点可以肯定的是,事务隔离级别越高,死锁出现的概率就越大。

为了提高性能,乐观锁(optimistic locking)机制被提出。与传统的悲观锁(optimistic locking)「先取锁再访问」的保守策略不同,乐观锁相信事物之间的数据竞争(data race)的概率是比较小的,因此尽可能直接做下去,直到提交的时候才去锁定。但如果直接简单这么做,还是有可能会遇到不可预期的结果,例如两个事务都读取了数据库的某一行,经过修改以后写回数据库,这时就遇到了问题。

一种可靠的乐观锁的实现是使用「多版本控制(multi-version control)」,即在每一行加一个version属性。修改这一行时将version增加1,写回数据库要检查当前的version值是否还是获取时的那个值了。如果还是,说明期间没有其他事务对其修改,直接提交即可,如果已经不是了,说明期间已经有别的事务修改了这一行,当前事务获取的数据已经过期了,事务失败。

在PostgreSQL及MySQL的InnoDB引擎的实现中,多版本控制的乐观锁是内置的(build-in),所以这个无需手动添加version字段。默认情况下,PostgreSQL/MySQL会尽可能使用乐观锁,除非遇到显式的锁定命令,如"select * from sometable for update"这样的语句,才会主动使用悲观锁。<o

参考资料

http://en.wikipedia.org/wiki/Isolation_(database_systems)http://www.blogjava.net/loocky/archive/2006/11/15/81138.html

这篇关于关系数据库的事务隔离、锁定与并发控制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/946808

相关文章

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

SpringKafka消息发布之KafkaTemplate与事务支持功能

《SpringKafka消息发布之KafkaTemplate与事务支持功能》通过本文介绍的基本用法、序列化选项、事务支持、错误处理和性能优化技术,开发者可以构建高效可靠的Kafka消息发布系统,事务支... 目录引言一、KafkaTemplate基础二、消息序列化三、事务支持机制四、错误处理与重试五、性能优

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

Python异步编程中asyncio.gather的并发控制详解

《Python异步编程中asyncio.gather的并发控制详解》在Python异步编程生态中,asyncio.gather是并发任务调度的核心工具,本文将通过实际场景和代码示例,展示如何结合信号量... 目录一、asyncio.gather的原始行为解析二、信号量控制法:给并发装上"节流阀"三、进阶控制

Redis中高并发读写性能的深度解析与优化

《Redis中高并发读写性能的深度解析与优化》Redis作为一款高性能的内存数据库,广泛应用于缓存、消息队列、实时统计等场景,本文将深入探讨Redis的读写并发能力,感兴趣的小伙伴可以了解下... 目录引言一、Redis 并发能力概述1.1 Redis 的读写性能1.2 影响 Redis 并发能力的因素二、

使用DrissionPage控制360浏览器的完美解决方案

《使用DrissionPage控制360浏览器的完美解决方案》在网页自动化领域,经常遇到需要保持登录状态、保留Cookie等场景,今天要分享的方案可以完美解决这个问题:使用DrissionPage直接... 目录完整代码引言为什么要使用已有用户数据?核心代码实现1. 导入必要模块2. 关键配置(重点!)3.

SpringSecurity 认证、注销、权限控制功能(注销、记住密码、自定义登入页)

《SpringSecurity认证、注销、权限控制功能(注销、记住密码、自定义登入页)》SpringSecurity是一个强大的Java框架,用于保护应用程序的安全性,它提供了一套全面的安全解决方案... 目录简介认识Spring Security“认证”(Authentication)“授权” (Auth

python之流程控制语句match-case详解

《python之流程控制语句match-case详解》:本文主要介绍python之流程控制语句match-case使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录match-case 语法详解与实战一、基础值匹配(类似 switch-case)二、数据结构解构匹

Seata之分布式事务问题及解决方案

《Seata之分布式事务问题及解决方案》:本文主要介绍Seata之分布式事务问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Seata–分布式事务解决方案简介同类产品对比环境搭建1.微服务2.SQL3.seata-server4.微服务配置事务模式1