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

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

相关文章

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络

通过Spring层面进行事务回滚的实现

《通过Spring层面进行事务回滚的实现》本文主要介绍了通过Spring层面进行事务回滚的实现,包括声明式事务和编程式事务,具有一定的参考价值,感兴趣的可以了解一下... 目录声明式事务回滚:1. 基础注解配置2. 指定回滚异常类型3. ​不回滚特殊场景编程式事务回滚:1. ​使用 TransactionT

使用Python实现一键隐藏屏幕并锁定输入

《使用Python实现一键隐藏屏幕并锁定输入》本文主要介绍了使用Python编写一个一键隐藏屏幕并锁定输入的黑科技程序,能够在指定热键触发后立即遮挡屏幕,并禁止一切键盘鼠标输入,这样就再也不用担心自己... 目录1. 概述2. 功能亮点3.代码实现4.使用方法5. 展示效果6. 代码优化与拓展7. 总结1.

用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的原始行为解析二、信号量控制法:给并发装上"节流阀"三、进阶控制