【MySQL】从零开始了解数据库开发 ---mysql事务机制(一)
从零开始了解数据库开发
- 1 mysql 事务是什么
- 2 为什么要有事务
- 3 事务的基本操作
1 mysql 事务是什么
事务是mysql的一个重要机制。对于数据库来说,是会被多个客户端访问的,假如对于一个火车票抢票系统:
北京到上海的车票还剩一张,但是有两个客户端同时看到余票一张,并决定抢票。
如果在A买完票,还没有执行更新数据库时,B同时也买票,那么不就造成了一张表被卖了两次吗!很显然,数据库的CRUD(增删查改)是必然满足一些性质的,以避免出现并发问题!
- 买票的过程必须是原子的,一个操作执行完,另外一个才能进行,不能存在中间状态。
- 买票的过程不能呼吸影响
- 买票后必须是持久有效的,不能说上一秒抢票成功,下一秒票就消失了
- 购买前与购买后的状态必须是确定的!
带着这些问题,我们来学习事务究竟是什么。
事务就是一组DML语句组成,这些语句在逻辑上存在相关性,这一组DML语句要么全部成功,要么全部失败,是一个整体。MySQL提供一种机制,保证我们达到这样的效果,事务还规定不同的客户端看到的数据是不相同的。
事务就是要做的或所做的事情,主要用于处理操作量大,复杂度高的数据。假设一种场景:你毕业了,学校的教务系统后台 MySQL 中,不在需要你的数据,要删除你的所有信息(一般不会 ), 那么要删除你的基本信息(姓名,电话,籍贯等)的同时,也删除和你有关的其他信息,比如:你的各科成绩,你在校表现,甚至你在论坛发过的文章等。这样,就需要多条 MySQL 语句构成,那么所有这些操作合起来,就构成了一个事务。
正如我们上面所说,一个 MySQL 数据库,不止一个事务在运行,同一时刻,甚至有大量的请求被包装成事务,在向 MySQL 服务器发起事务处理请求。而每条事务至少一条 SQL ,这样如果大家都访问同样的表数据,在不加保护的情况,就绝对会出现问题。甚至,因为事务由多条 SQL 构成,那么,也会存在执行到一半出错或者不想再执行的情况,那么已经执行的怎么办呢?
因此,一个完整的事务,绝对不是简单的 sql 集合,还需要满足如下四个属性ACID:
- 原子性(Atomicity):一个事务(transaction)中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。事务在执行过程中发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。
- 一致性(Consistency):在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的资料必须完全符合所有的预设规则,这包含资料的精确度、串联性以及后续数据库可以自发性地完成预定的工作。
- 隔离性(Isolation):数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。事务隔离分为不同级别,包括读未提交( Readuncommitted )、读提交( read committed )、可重复读( repeatable read )和串行化( Serializable )
- 持久性(Durability):事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失。
2 为什么要有事务
事务被 MySQL 编写者设计出来,事务不是天然就有的(数据库的增删查改是天然存在的),本质是为了当应用程序访问数据库的时候,事务能够简化我们的编程模型,不需要我们去考虑各种各样的潜在错误和并发问题。
可以想一下当我们使用事务时,要么提交,要么回滚,我们不会去考虑网络异常了,服务器宕机了,同时更改一个数据怎么办,因此事务本质上是为了应用层服务的。而不是伴随着数据库系统天生就有的。
3 事务的基本操作
在 MySQL 中只有使用了 Innodb 数据库引擎的数据库或表才支持事务, MyISAM 不支持!
事务提交的方式一般有两种:手动提交,自动提交。
是否自动提交可以查看:
mysql> show variables like 'autocommit';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit | ON |
+---------------+-------+
1 row in set (0.41 sec)
mysql> SET AUTOCOMMIT=0; # SET AUTOCOMMIT=0 禁止自动提交、 =1 自动提交
Query OK, 0 rows affected (0.00 sec)
mysql> show variables like 'autocommit';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit | OFF |
+---------------+-------+
1 row in set (0.00 sec)
接下来我们建立一下测试用的表结构以及测试环境
## 为了便于演示,我们将mysql的默认隔离级别设置成读未提交。
## 具体操作我们后面专门会讲,现在已使用为主。
mysql> set global transaction isolation level READ UNCOMMITTED;
Query OK, 0 rows affected (0.00 sec)
mysql> quit
Bye
##需要重启终端,进行查看
mysql> select @@transaction_isolation;
+-------------------------+
| @@transaction_isolation |
+-------------------------+
| READ-UNCOMMITTED |
+-------------------------+
1 row in set (0.00 sec)
创建测试表结构:
mysql> create table if not exists account( id int primary key, name varchar(50) not null default '', blance decimal(10,2) not null default 0.0 )ENGINE=InnoDB DEFAULT CHARSET=UTF8;
Query OK, 0 rows affected, 1 warning (0.03 sec)
接下来我们就来操作事务:
- 开始一个事务begin也可以,推荐begin:
start transaction;
BEGIN;
- 创建一个保存点save1,
savepoint save1;
- 回滚在最开始 rollback;
rollback
,只能在事务中进行回滚 - 回滚到指定保存点:
rollback to save1
,只能在事务中进行回滚 - 提交事务:
commit;
接下来我们启动事务进行测试
接下来就可以在测试事务的操作了。
因为是rc隔离级,当我们左边插入时,右边直接查询就能搜到新纪录。
当需要结束事务,进行commit提交就可以,就持久化的保存到数据库中无法进行回滚了!
对于一些特殊场景:
- 当事务中途异常时,比如客户端崩溃,断开连接之类的,会如何呢?客户端崩溃,MySQL自动会回滚(隔离级别设置为读未提交)
- 事务已经commit了,客户端崩溃,MySQL数据不会在受影响,已经持久化
- begin操作会自动更改提交方式,不会受MySQL是否自动提交影响
平时单条sql与事务又是什么关系呢?
- 简单来说,在 MySQL 默认的自动提交(autocommit)模式下,每一条单独的 SQL 语句(DML/DDL)本身就是一个完整的事务
- 在 MySQL 设置非自动提交模式下,你执行的每一条 SQL 语句,都会默认成为当前事务的一部分,而不会自动提交,需要手动提交!
结论:
- 只要输入begin或者start transaction,事务便必须要通过commit提交,才会持久化,与是否设置set autocommit无关。
- 事务可以手动回滚,同时,当操作异常,MySQL会自动回滚
- 对于 InnoDB 每一条 SQL 语言都默认封装成事务,自动提交。(select有特殊情况,因为MySQL 有 MVCC )
- 从上面的例子,我们能看到事务本身的原子性(回滚),持久性(commit)
- 那么隔离性?一致性?我们后面来看
事务操作注意事项:
- 如果没有设置保存点,也可以回滚,只能回滚到事务的开始。直接使用 rollback(前提是事务还没有提交)
- 如果一个事务被提交了(commit),则不可以回退(rollback)
- 可以选择回退到哪个保存点
- InnoDB 支持事务, MyISAM 不支持事务
- 开始事务可以使 start transaction 或者 begin