【颗粒归仓】--数据库事务

2024-08-26 02:38

本文主要是介绍【颗粒归仓】--数据库事务,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

   说到事务,做IT的肯定都不陌生,会想到Transcation(开启、提交、回滚)。作为资深的IT工作者,对事务全面理解是一项基本功,当然对于正在迈向资深的我来说,也必须了解。下面就来谈一下事务

 一、事务的概念

   数据库事务(Database Transaction) ,是指作为单个逻辑工作单元执行的一系列操作,要么完全地执行,要么完全地不执行。 


 二、事务的特性

  事务必须具备ACID四个特性

  原子性(Atomicity): 一个事务的执行被视为一个不可分割的单元,事务中的操作,要么全部成功,要么全部失败回滚,不能只执行其中的一部分。

  一致性(Consistency):一致性是指事务必须使数据库从一个一致的状态变到另一个一致的状态,也就是执行事务的前后状态都必须处于一致的状态。比如说A向B转账500,那么A账户要减500,而B账户需要加500,这两个事件最终的账户总额仍然没有改变。

  隔离性(Isolation):是指事务之间的行为互不影响,在实际操作中,会受事务的隔离级别的影响。

  持久性(Durability):事务提交之后,将数据持久到数据库,即使系统出现问题,数据依然还在。


 三、事务的隔离级别

  在说到事务的隔离特性时,提到事务的隔离性会受到隔离级别的影响。事务的隔离级别定义了事务之间的可见性,有人称它为事务的“自私”程度。主要分为四种隔离级别

  1、READ UNCOMMITTED(未提交读)。

     在此隔离级别下,如果事务A对数据进行修改,在没有提交的情况下,事务B也可以看到该修改,这种问题称为脏读。未提交读的隔离级别较低,容易引起很多问题,一般不常用。

     测试如下:(使用navicat连接mysql进行测试)

     ①首先要将隔离级别设置为未提交读状态(以下实例都在global级别设置,也可以设置session级别)

      打开第一个命令行界面,用来专门设置隔离级别:

      mysql> set @@global.tx_transaction='READ-UNCOMMITTED';

      因为设置的是global级别的,所以在接下来的操作中,为了能更好的演示效果,需要另开一个命令行窗口,在接下来的描述中,提到创建事务就表示新打开一个窗口。

      ②创建A事务:

        首先要查看当前的隔离级别,确保是未提交读状态:

        

        开启事务,并执行查询,可以看到只有三条数据。结果如下:

        

       ③创建B事务:

         也是先查看当前的隔离级别(以下测试都会在新打开命令行界面时查看隔离级别,后面就不再叙述)

         开启事务,并插入一条数据,但是不提交。此时,在当前事务中可以查询到新增的数据:

         

       ④然后再回到A事务,执行查询,此时在B事务还未提交时,就可以查到新增的数据:

       

      在A事务中读到了B事务未提交的数据,称此现象为脏读。


  2、READ COMMITTED(提交读)

    在该隔离级别下,不会出现脏读。当事务A对数据修改之后,只要在提交以后,事务B才能看到修改的数据。但是会出现不可重复读问题,何谓不可重复读?就是一个事务对同一行数据重复读取两次,但是却得到了不同的结果。

    测试如下:

     ①设置隔离级别

       mysql> set @@global.tx_isolation='READ-COMMITTED';

     ②创建事务A:

       开启事务,然后查询数据

       

      ③创建事务B

        开启事务,接下来查看数据,跟事务A查到的数据一致。然后更新第二条数据。此时,不提交事务。

        

       ④在事务A中执行查询命令,发现B事务更新的数据查不到:

       

       ⑤让事务B提交,然后事务A再次执行查询,此时就可以查看到更新之后的数据:

       

     在该隔离级别下,事务A读到的内容是事务B提交后的数据,即事务A只能读到事务B两端的数据,至于事务B中间的其他修改就读不到了。如事务A能读到刚开始的状态,如果事务B在修改过程中,刚开始把id=10的数据修改为‘cc’,没有提交;然后又把id=11的数据改为‘dd’,依旧不提交;最后发现更新错了,又把id=10的数据改为‘c’,id=11的数据改为‘d’,把id=9的数据改为‘bb’,提交。此时事务A只能读到id=9的值变了,至于id=10和11的变化过程一无所知。这个就是与未提交读的区别之处,未提交读能看到这个变化过程,随变化而变化。

  3、REPEATABLE  READ(可重复读)

     在该隔离级别下,不会出现不可重复读的问题。事务B对数据做的修改,提交之后,对于先于B事务开启的事务A来说,此修改仍然看不到。

     测试如下:

      ①设置隔离级别

        mysql> set@@global.tx_isolation='REPEATABLE-READ';

      ②创建事务A

        开启事务,然后执行查询

        

      ③创建事务B

        开启事务B,先查询数据,然后将第三条数据更新。此时不提交事务

        

     ④在事务A中进行查询,无法查看事务B修改的数据

     ⑤提交事务B,再在事务A中执行查询,依然无法看到更新的数据,查询到的数据始终没有改变

     

     ⑥将事务A提交,然后开启新的事务,此时即可查看到更新后的数据


   可重复读解决了不可重复读的问题,某个事务在执行期间,无论后续的其他事务如何操作数据,都保持查询结果不变,但是重新打开事务后,会读到不一样的数据。这就引出了幻读的问题,所问幻读就是事务在操作过程中进行两次查询,第二次查询的结果包含了第一次查询中未出现的数据或者缺少了第一次查询中出现的数据。


  4、SERIALIZABLE(可串行化)

   这是隔离的最高级别,强制要求所有的事务串行执行,会对每行数据加锁,这就造成大量锁征用问题,性能最差。

   测试如下:

    ①设置隔离级别

       mysql>  set @@global.tx_isolation='SERIALIZABLE';

    ②创建事务A

      开启事务并执行查询

      

    ③创建事务B

      开启事务并执行查询,此时可以查询。但是如果更新数据时,事务处于等待状态,因为事务A已经上锁。当一定时间还无法执行时,提示当前更新数据处于锁状态,无法更新。

      

     ④当事务A提交后,处于等待状态的事务B也会执行完。

     

    可串行化解决了幻读的问题,但是又引发了新的问题,死锁。例如上述两个事务,如果都不提交,那么此时都将无法进行后续操作。


   综合上述内容,四种隔离级别所解决的问题如下:

   

   其性能如下:

   

     即隔离级别越高,其并发性越低,性能越差。


  四、小结

    事务是开发中无法避免的一个问题,所以熟悉事务的基本概念以及使用是非常重要的。当然本文也只是解析了事务的一角,还需要在今后的工作中继续深入剖析事务的应用。

  

这篇关于【颗粒归仓】--数据库事务的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

深入理解数据库的 4NF:多值依赖与消除数据异常

在数据库设计中, "范式" 是一个常常被提到的重要概念。许多初学者在学习数据库设计时,经常听到第一范式(1NF)、第二范式(2NF)、第三范式(3NF)以及 BCNF(Boyce-Codd范式)。这些范式都旨在通过消除数据冗余和异常来优化数据库结构。然而,当我们谈到 4NF(第四范式)时,事情变得更加复杂。本文将带你深入了解 多值依赖 和 4NF,帮助你在数据库设计中消除更高级别的异常。 什么是

DM8数据库安装后配置

1 前言 在上篇文章中,我们已经成功将库装好。在安装完成后,为了能够更好地满足应用需求和保障系统的安全稳定运行,通常需要进行一些基本的配置。下面是一些常见的配置项: 数据库服务注册:默认包含14个功能模块,将这些模块注册成服务后,可以更好的启动和管理这些功能;基本的实例参数配置:契合应用场景和发挥系统的最大性能;备份:有备无患;… 2 注册实例服务 注册了实例服务后,可以使用系统服务管理,

速了解MySQL 数据库不同存储引擎

快速了解MySQL 数据库不同存储引擎 MySQL 提供了多种存储引擎,每种存储引擎都有其特定的特性和适用场景。了解这些存储引擎的特性,有助于在设计数据库时做出合理的选择。以下是 MySQL 中几种常用存储引擎的详细介绍。 1. InnoDB 特点: 事务支持:InnoDB 是一个支持 ACID(原子性、一致性、隔离性、持久性)事务的存储引擎。行级锁:使用行级锁来提高并发性,减少锁竞争

MySql 事务练习

事务(transaction) -- 事务 transaction-- 事务是一组操作的集合,是一个不可分割的工作单位,事务会将所有的操作作为一个整体一起向系统提交或撤销请求-- 事务的操作要么同时成功,要么同时失败-- MySql的事务默认是自动提交的,当执行一个DML语句,MySql会立即自动隐式提交事务-- 常见案例:银行转账-- 逻辑:A给B转账1000:1.查询

开源分布式数据库中间件

转自:https://www.csdn.net/article/2015-07-16/2825228 MyCat:开源分布式数据库中间件 为什么需要MyCat? 虽然云计算时代,传统数据库存在着先天性的弊端,但是NoSQL数据库又无法将其替代。如果传统数据易于扩展,可切分,就可以避免单机(单库)的性能缺陷。 MyCat的目标就是:低成本地将现有的单机数据库和应用平滑迁移到“云”端

ORACLE 11g 创建数据库时 Enterprise Manager配置失败的解决办法 无法打开OEM的解决办法

在win7 64位系统下安装oracle11g,在使用Database configuration Assistant创建数据库时,在创建到85%的时候报错,错误如下: 解决办法: 在listener.ora中增加对BlueAeri-PC或ip地址的侦听,具体步骤如下: 1.启动Net Manager,在“监听程序”--Listener下添加一个地址,主机名写计

MyBatis 切换不同的类型数据库方案

下属案例例当前结合SpringBoot 配置进行讲解。 背景: 实现一个工程里面在部署阶段支持切换不同类型数据库支持。 方案一 数据源配置 关键代码(是什么数据库,该怎么配就怎么配) spring:datasource:name: test# 使用druid数据源type: com.alibaba.druid.pool.DruidDataSource# @需要修改 数据库连接及驱动u

Lua 脚本在 Redis 中执行时的原子性以及与redis的事务的区别

在 Redis 中,Lua 脚本具有原子性是因为 Redis 保证在执行脚本时,脚本中的所有操作都会被当作一个不可分割的整体。具体来说,Redis 使用单线程的执行模型来处理命令,因此当 Lua 脚本在 Redis 中执行时,不会有其他命令打断脚本的执行过程。脚本中的所有操作都将连续执行,直到脚本执行完成后,Redis 才会继续处理其他客户端的请求。 Lua 脚本在 Redis 中原子性的原因