MySQL事务的四种隔离类型以及PHP框架Yii2中的源码解读和实际应用

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: MySQL事务的四种隔离类型以及PHP框架Yii2中的源码解读和实际应用

一、什么是事务

事务(Transaction) 是并发控制的基本单位。所谓的事务,它是一个操作序列,这些操作要么都 执行,要么都不执行,它是一个不可分割的工作单位。事务是数据库维护数据一致性的单位,在每 个事务结束时,都能保持数据一致性。

二、事务的特性

ACID,分别是原子性、一致性、隔离性和持久性。

1. 原子性 Atomicity

一个事务(transaction)中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。事务在执行过程中发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。

2. 一致性 Consistency

在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的资料必须完全符合所有的预设规则,这包含资料的精确度、串联性以及后续数据库可以自发性地完成预定的工作。

3. 隔离性

数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。事务隔离分为不同级别,包括读未提交(Read uncommitted)、读提交(read committed)、可重复读(repeatable read)和串行化(Serializable)。

4. 持久性

事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失。

三、事务的隔离级别

事务的隔离级别的严格程度从上到下依次为

  1. 读未提交(Read uncommitted)
  2. 读提交(read committed)
  3. 可重复读(repeatable read)
  4. 串行化(Serializable)

1.读未提交(Read uncommitted)

一个事务读取到其他事务未提交的数据,是级别最低的隔离机制。

2.读提交(read committed)

一个事务读取到其他事务提交后的数据

3.可重复读(repeatable read)

一个事务对同一份数据读取到的相同,不在乎其他事务对数据的修改

4. 串行化(Serializable)

事务串行化执行,隔离级别最高,牺牲了系统的并发性

五、因为事务可能出现的问题

1. 脏读

事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据。

2. 不可重复读

事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果 不一致。

3. 幻读

同一事务中对同一范围的数据进行读取,结果却多出了数据或者少了数据,这就叫幻读。(如同一事务对id<10的范围进行2次查询,第一次出现id=8、9的两条数据,第二次出现id=7、8、9的3条数据)。

六、四种隔离级别对脏读、不可重复读、幻读的影响

脏读 不可重复读 幻读
读未提交(Read uncommitted) * * *
读提交(read committed) * *
可重复读(repeatable read) *
串行化(Serializable)

七、Yii2中如何使用事务

1. 源码位置和解读

Yii2的事务相关代码的位置是vendor/yiisoft/yii2/db/Connection.php

/**
     * Starts a transaction.
     * @param string|null $isolationLevel The isolation level to use for this transaction.
     * See [[Transaction::begin()]] for details.
     * @return Transaction the transaction initiated
     */
    public function beginTransaction($isolationLevel = null)
    {
        $this->open();
        if (($transaction = $this->getTransaction()) === null) {
            $transaction = $this->_transaction = new Transaction(['db' => $this]);
        }
        $transaction->begin($isolationLevel);
        return $transaction;
    }

其中设定事务隔离类型的方法如下

/**
     * Sets the isolation level of the current transaction.
     * @param string $level The transaction isolation level to use for this transaction.
     * This can be one of [[Transaction::READ_UNCOMMITTED]], [[Transaction::READ_COMMITTED]], [[Transaction::REPEATABLE_READ]]
     * and [[Transaction::SERIALIZABLE]] but also a string containing DBMS specific syntax to be used
     * after `SET TRANSACTION ISOLATION LEVEL`.
     * @see http://en.wikipedia.org/wiki/Isolation_%28database_systems%29#Isolation_levels
     */
    public function setTransactionIsolationLevel($level)
    {
        $this->db->createCommand("SET TRANSACTION ISOLATION LEVEL $level")->execute();
    }

通过注释我们可以发现四种隔离类型分别通过常量设置

  • Transaction::READ_UNCOMMITTED
  • Transaction::READ_COMMITTED
  • Transaction::REPEATABLE_READ
  • Transaction::SERIALIZABLE

常量所在位置vendor/yiisoft/yii2/db/Transaction.php

/**
     * A constant representing the transaction isolation level `READ UNCOMMITTED`.
     * @see http://en.wikipedia.org/wiki/Isolation_%28database_systems%29#Isolation_levels
     */
    const READ_UNCOMMITTED = 'READ UNCOMMITTED';
    /**
     * A constant representing the transaction isolation level `READ COMMITTED`.
     * @see http://en.wikipedia.org/wiki/Isolation_%28database_systems%29#Isolation_levels
     */
    const READ_COMMITTED = 'READ COMMITTED';
    /**
     * A constant representing the transaction isolation level `REPEATABLE READ`.
     * @see http://en.wikipedia.org/wiki/Isolation_%28database_systems%29#Isolation_levels
     */
    const REPEATABLE_READ = 'REPEATABLE READ';
    /**
     * A constant representing the transaction isolation level `SERIALIZABLE`.
     * @see http://en.wikipedia.org/wiki/Isolation_%28database_systems%29#Isolation_levels
     */
    const SERIALIZABLE = 'SERIALIZABLE';

2. 业务逻辑中开始事务

先上代码,如下是一般业务开启事务的基本用法。

$transaction = Yii::$app->db->beginTransaction();
        try {
          //这里进行对model的读写操作
          $transaction->commit();
            return true;
        }catch(\Exception $e) {
            $transaction->rollBack();
            return false;
        }

基本逻辑为

  1. 开启事务
  2. try catch 对异常进行捕获
  3. 如果没有捕获异常就在最后调用提交$transaction->commit();
  4. 如果捕获异常就回滚$transaction->rollBack();

八、总结

开发人员对事务的掌握和理解是必须的,并且在复杂的业务中也必须掌握事务的用法,以此来保证业务数据的一致性。

相关实践学习
基于CentOS快速搭建LAMP环境
本教程介绍如何搭建LAMP环境,其中LAMP分别代表Linux、Apache、MySQL和PHP。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
5天前
|
SQL 存储 关系型数据库
MySQL索引及事务
MySQL索引及事务
17 2
|
1天前
|
Java 数据挖掘 BI
Java医院绩效考核系统源码B/S+avue+MySQL助力医院实现精细化管理
医院绩效考核系统目标是实现对科室、病区财务指标、客户指标、流程指标、成长指标的全面考核、分析,并与奖金分配、学科建设水平评价挂钩。
30 0
|
2天前
|
安全 编译器 API
深入理解PHP7中的返回类型声明
【5月更文挑战第12天】 在面向对象编程中,明确函数和方法是关键。而PHP7引入的返回类型声明特性,为开发者提供了一种强大的工具来增强代码的清晰度和可靠性。本文将深入探讨PHP7返回类型声明的概念、用法以及它如何帮助我们编写更加健壮和易于维护的代码。
|
4天前
|
PHP 开发者
深入理解PHP7的返回类型声明
【5月更文挑战第11天】 在PHP 7中,引入了一项新的语言特性 - 返回类型声明。这一特性允许开发者在函数定义时指定预期的返回值类型,从而增强代码的可读性和健壮性。本文将探讨返回类型声明的概念、语法、优势以及在实际开发中的应用,帮助读者更深入地理解并有效运用这一新特性。
|
5天前
|
存储 算法 关系型数据库
MySQL事务与锁,看这一篇就够了!
MySQL事务与锁,看这一篇就够了!
|
6天前
|
Java 关系型数据库 MySQL
MySQL 索引事务
MySQL 索引事务
13 0
|
7天前
|
存储 关系型数据库 MySQL
9.2.5.2 【MySQL】XDES 类型
9.2.5.2 【MySQL】XDES 类型
11 0
9.2.5.2 【MySQL】XDES 类型
|
7天前
|
传感器 人工智能 前端开发
JAVA语言VUE2+Spring boot+MySQL开发的智慧校园系统源码(电子班牌可人脸识别)Saas 模式
智慧校园电子班牌,坐落于班级的门口,适合于各类型学校的场景应用,班级学校日常内容更新可由班级自行管理,也可由学校统一管理。让我们一起看看,电子班牌有哪些功能呢?
100 4
JAVA语言VUE2+Spring boot+MySQL开发的智慧校园系统源码(电子班牌可人脸识别)Saas 模式
|
8天前
|
存储 关系型数据库 MySQL
MySQL字段的字符类型该如何选择?千万数据下varchar和char性能竟然相差30%🚀
本篇文章来讨论MySQL字段的字符类型选择并深入实践char与varchar类型的区别以及在千万数据下的性能测试
MySQL字段的字符类型该如何选择?千万数据下varchar和char性能竟然相差30%🚀
|
13天前
|
SQL 安全 关系型数据库
【Mysql-12】一文解读【事务】-【基本操作/四大特性/并发事务问题/事务隔离级别】
【Mysql-12】一文解读【事务】-【基本操作/四大特性/并发事务问题/事务隔离级别】