搜尋
首頁資料庫mysql教程并发操作与数据的不一致性

最近做的《选修课系统》需要考虑这样一个问题:数据库的并发操作带来的数据库数据不一致问题因为是全校性选修课,同一时间点大批学生选课,那么必然存在多名学生同时对同一数据进行操作是的问题,如果这种并发操作不加以控制的话,必会造成数据的不一致。 一

最近做的《选修课系统》需要考虑这样一个问题:数据库的并发操作带来的数据库数据不一致问题——因为是全校性选修课,同一时间点大批学生选课,那么必然存在多名学生同时对同一数据进行操作是的问题,如果这种并发操作不加以控制的话,必会造成数据的不一致。

一直知道有这种问题,并且知道这种问题的解决方法——加锁;但是有些东西之前了解的并不是很透彻,于是好好研究了一天,觉得理解的还可以,先总结一部分。

1,什么是并发操作?

数据库的一个重要特征是:支持数据共享,也就是说允许多个用户程序并行地存取数据库中的数据;那么,多用户或多事物可能同时对同一数据进行操作,这成为并发操作。

2,并发操作可能带来的影响?

如果不对并发操作进行控制的话,那么就会存取不正确的数据,破坏数据的完整性。——为什么这么说?下面进行介绍

在此之前,先说一下事务的概念。

3,什么是事务?

对一件完整的事儿,要么做完整,要么都不做。例如:学生选课—学生选上课时,需要将选课信息写入选课表的同时,更新课程表中的课程余量;如果在添加选课信息时,中断了(不能进行余量更新),那么就会事务回滚,即,数据回滚到没选课之前。

Begin Transaction

选课表中添加选课信息;

更新课程表中的课程余量;

Commit

RollBac

事务的SQL语句:

Begin Transaction开始事务

Commit提交事务

RollBack回滚

注:事务中的多条SQL语句也是一条条执行的,当所有语句执行完后,提交事务,如果其中一条中断,则事务回滚所有操作回到语句执行之前)

事务的并发主要是为了提高效率,但是,同时它也带来了一定问题

4,并发操作带来的问题?

之前讲过并发操作——多用户或多事务同时对同一数据进行操作;

因为事务中的语句也是一条条执行的,所以存在多用户多事务同时对同一数据进行操作的情况;

并发操作带来的问题:

(1)丢失修改

(2)脏读

(3)不可重复读

4.1丢失修改

当两个或多个事务(或两个或多个用户)选择同一行,然后基于最初选定的值更新该行时,会发生丢失更新问题。每个事务都不知道其它事务的存在(或每个用户操作时并不会考虑同一时刻是否有别的用户进行着同样的操作)。最后的更新将重写由其它事务所做的更新,这将导致数据丢失。  

例子1:事务T1,事务T2,数据库中数据R=1000

a.t1时刻,事务T1读取R=1000;

b.过了一会儿 t2时刻,事务T2读取R=1000;

c.t3时刻,事务T1修改R=R-200(那么R=800)写入数据库(此时,数据库中R=800);

d.过了一会儿 t4时刻,事务T2修改R=R-100(因为在t2时刻读到的数据为R=1000,那么修改后R=900)写入数据库(此时,数据库中R=900)

那么最终,数据库中R=900;数据对吗?当然不对,因为T1、T2分别对R进行了-200 -100操作,最终数据应为R=700;事务T2的修改覆盖了T1的修改 ——丢失修改问题

例子2:选课:学生A、学生B、课程1的余量=20

a.学生A选择课程1时,先读出课程余量20

b.然而同一时刻(也可是不同时刻,只要在学生A更新数据之前),学生B也读出了课程1的余量20,

c.学生A选择此课程,课程1余量-1,写入数据库,此时课程1的余量=19;

d.学生B选择此课程,课程1余量-1(因为之前读出的课程1余量为20,-1后为19),写入数据库,此时课程1的余量=19;

课程余量=19 数据正确吗?不正确。实际情况课程余量应更新为18;——B的修改覆盖了A的修改 —— 丢失修改问题。

解决办法:加锁,只允许并发一个更新事务。

4.2脏读

当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,

另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据是还没有提交的数据,那么另外一个

事务读到的这个数据是脏数据,依据脏数据所做的操作可能是不正确的。

例子1:事务T1,事务T2,数据库中数据R=1000;

T1:

Begin Transaction开始事务T1

ReadR=1000;(1)

R=R-200;(2)此时R=800

R=R+100;(3)此时R=900

Commit提交事务T1

a.当事务T1进行到第(2)后,即,数据库中的数据R=800时;

b.事务T2开始读R的值,读出的值为R=800;但此时事务T1还没有进行完整,还未提交事务;

c.之后事务T1进行第(3)步R=R+100,此时R=900 事务T1提交,此时数据库中R=900;

那么在事务T1提交事务之前,事务T2读出的数据(R=800)为脏数据;

例子2:

a.张三的工资为2000,元老板把张三的工资改为了8000元(但未提交事务)

2.张三查看自己的工资 ,发现工资变为了8000元

3.而后老板发现改错了,回滚了事务,张三的工资又变回了2000元

那么,张三读取的工资8000元就是脏数据。

解决办法:如果在第一个事务提交前,任何其他事务不可读取其修改过的值,则可以避免该问题。

4.3不可重复读

  当第二个事务多次访问同一行而且每次读取不同的数据时,会发生不一致的分析问题。不一致的分析与未确认的相关性类似,因为其它事务也是正在更改第二个事务正在读取的数据。然而,在不一致的分析中,第二个事务读取的数据是由已进行了更改的事务提交的。而且,不一致的分析涉及多次(两次或更多)读取同一行,而且每次信息都由其它事务更改;因而该行被非重复读取。

在一个事务中前后两次读取的结果并不致,导致了不可重复读。

例子1:事务T1、事务T2、张三的工资=1000

a.事务T1中,张三读取了自己的工资为1000元,操作并没有完成

b.此时(事务T1读取了张三工资为1000元)事务2中,修改了张三的工资为2000元,并提交了事务.

c.此时(事务T1读取了张三工资为1000元,事务T2修改了张三的工资为2000元)在事务1中,张三再次读取自己的工资时,工资变为了2000

那么,同一个事务中,前后读取的数据不一致 —— 不可重复读问题。

解决办法:如果只有在修改事务完全提交之后才可以读取数据,则可以避免该问题。

总结:

(1)事务的并发主要是为了提高效率,但是,同时它也带来了一定问题——丢失修改、读脏数据、不可重复读

(2)结合生活中实例,理解丢失修改、读脏数据、不可重复读问题。

个人理解,如果哪里有理解偏差,忘纠正!

下篇博客会总结“锁”的概念,并解释“锁”是如何解决并发造成的数据不一致问题。

陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
Go 语言中的 goroutine 是什么?Go 语言中的 goroutine 是什么?Jun 11, 2023 am 11:50 AM

Go语言是一种开源编程语言,由Google开发并于2009年面世。这种语言在近年来越发受到关注,并被广泛用于开发网络服务、云计算等领域。Go语言最具特色的特点之一是它内置了goroutine(协程),这是一种轻量级的线程,可以在代码中方便地实现并发和并行计算。那么goroutine到底是什么呢?简单来说,goroutine就是Go语言中的

Java 中的锁机制Java 中的锁机制Jun 08, 2023 am 08:03 AM

Java作为一种高级编程语言,在并发编程中有着广泛的应用。在多线程环境下,为了保证数据的正确性和一致性,Java采用了锁机制。本文将从锁的概念、类型、实现方式和使用场景等方面对Java中的锁机制进行探讨。一、锁的概念锁是一种同步机制,用于控制多个线程之间对共享资源的访问。在多线程环境下,线程的执行是并发的,多个线程可能会同时修改同一数据,这就会导致数

如何解决Python的函数中的并发不安全错误?如何解决Python的函数中的并发不安全错误?Jun 24, 2023 pm 12:37 PM

Python是一门流行的高级编程语言,它具有简单易懂的语法、丰富的标准库和开源社区的支持,而且还支持多种编程范式,例如面向对象编程、函数式编程等。尤其是Python在数据处理、机器学习、科学计算等领域有着广泛的应用。然而,在多线程或多进程编程中,Python也存在一些问题。其中之一就是并发不安全。本文将从以下几个方面介绍如何解决Python的函数中的并发不安

PHP8.0如何使用Fibers实现并发PHP8.0如何使用Fibers实现并发May 14, 2023 am 09:01 AM

随着现代互联网技术的不断发展,网站访问量越来越大,对于服务器的并发处理能力也提出了更高的要求。如何提高服务器的并发处理能力是每个开发者需要面对的问题。在这个背景下,PHP8.0引入了Fibers这一全新的特性,让PHP开发者掌握一种全新的并发处理方式。Fibers是什么?首先,我们需要了解什么是Fibers。Fibers是一种轻量级的线程,可以高效地支持PH

Java的并发异常——java.util.ConcurrentModificationException怎么办?Java的并发异常——java.util.ConcurrentModificationException怎么办?Jun 25, 2023 am 11:46 AM

Java作为一种高级语言,在编程语言中使用广泛。在Java的应用程序和框架的开发中,我们经常会碰到并发的问题。并发问题是指当多个线程同时对同一个对象进行操作时,会产生一些意想不到的结果,这些问题称为并发问题。其中的一个常见的异常就是java.util.ConcurrentModificationException异常,那么我们在开发过程中如何有效地解决这个异

Java中如何使用ConcurrentLinkedQueue函数进行并发队列操作Java中如何使用ConcurrentLinkedQueue函数进行并发队列操作Jun 26, 2023 pm 05:37 PM

Java中的ConcurrentLinkedQueue函数为开发者提供了一种线程安全的、高效的队列实现方式,它支持并发读写操作,并且执行效率较高。在本文中,我们将介绍Java中如何使用ConcurrentLinkedQueue函数进行并发队列操作,帮助开发者更好地利用其优势。ConcurrentLinkedQueue是Java中的一个线程安全、非阻塞的队列实

使用Go和Goroutines实现高效的并发图计算使用Go和Goroutines实现高效的并发图计算Jul 21, 2023 pm 03:58 PM

使用Go和Goroutines实现高效的并发图计算引言:随着大数据时代的到来,图计算问题也成为了一个热门的研究领域。在图计算中,图的顶点和边之间的关系非常复杂,因此如果采用传统的串行方法进行计算,往往会遇到性能瓶颈。为了提高计算效率,我们可以利用并发编程的方法使用多个线程同时进行计算。今天我将向大家介绍使用Go和Goroutines实现高效的并发图计算的方法

PHP和WebDriver扩展:如何模拟多个用户的并发访问PHP和WebDriver扩展:如何模拟多个用户的并发访问Jul 07, 2023 pm 06:03 PM

PHP和WebDriver扩展:如何模拟多个用户的并发访问随着互联网的快速发展,网站的访问量也越来越大,很多场景下需要测试网站在高并发情况下的表现。本文将介绍如何使用PHP和WebDriver扩展来模拟多个用户的并发访问,并提供相应的代码示例。首先,我们需要安装并配置PHP和WebDriver扩展。PHP是一种流行的服务器端脚本语言,而WebDriver是一

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
2 週前By尊渡假赌尊渡假赌尊渡假赌
倉庫:如何復興隊友
4 週前By尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒險:如何獲得巨型種子
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

MantisBT

MantisBT

Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

PhpStorm Mac 版本

PhpStorm Mac 版本

最新(2018.2.1 )專業的PHP整合開發工具

VSCode Windows 64位元 下載

VSCode Windows 64位元 下載

微軟推出的免費、功能強大的一款IDE編輯器

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

Atom編輯器mac版下載

Atom編輯器mac版下載

最受歡迎的的開源編輯器