百分百源码网-让建站变得如此简单! 登录 注册 签到领金币!

主页 | 如何升级VIP | TAG标签

当前位置: 主页>网站教程>数据库> MySQL 事务最全详解
分享文章到:

MySQL 事务最全详解

发布时间:09/01 来源:未知 浏览: 关键词:
什么是事务?

用 MySQL 官方的一句话来描写事务是啥?MySQL 事务主要用于处置操纵量大,复杂度高的数据。那何为数据量大?何为复杂度高呢?我用我本人的懂得来描写一下吧。事务其实就是 MySQL 中处置数据的一种方式,主要用在数据完全性高,数据之间依靠性大的状况下的一种数据处置方式。举个例子,小张向小李的银行卡打 200 块钱,在小张点击了确定转账的按钮时,系统忽然崩溃了。会显现这样几中不准确的状况:

1. 小张的钱打到小李的账户上,但是本人的账户上的钱没被扣.

2. 小张的钱打没到小李的账户上了,但是本人账户上的钱被扣.

这样的业务场景就需要 MySQL 事务保持,即便机器出故障的状况下,数据依然是准确的.

事务使用的前提

MySQL 要使用事务,需要 MySQL 中的储备引擎支撑。现当前 MySQL 内置的储备引擎支撑事务的有 InnoDB、NDB cluster, 第三方的储备引擎有 PBXT 和 XtrDB.

事务有什么特点?

MySQL 中的事务有如下几个特点 (ACID):

原子性 (atomicity):

一个事务必需被作为一个不成分割的最小工作单元,每个事务中的所有操纵必需要末成功,或者要末失败,永久不成能一些操纵失败,一些操纵成功,这就是所谓的原子性的概念.

一致性 (consistency):

一致性就像上面举的一个例子一样,当发生非常状况下,数据依然是准确的。就是说当一个事务施行失败了,数据之间是不会受非常的状况而影响,永久保持着他的准确性.

隔离性 (isolation):

当一个事务还未提交,每个事务之间是彼此隔离的,互补受到影响.

耐久性 (durability):

当一个事务停止提交之后,发生的转变就会永久留存在数据库中.

事务的隔离级别

在谈及到 MySQL 的隔离性的特点,就不得不说说隔离性的几种级别。至于为什么会触及到这一点,可以这样简便的懂得:假如统一时刻,有两个恳求在施行事务的操纵,并且这两个事务是对统一条数据做操纵,那么到底终究的结果是以谁的为准呢?不一样的隔离级别致使的结果不一样,因此事务的隔离级别也是一个非常重要的点.

隔离级别分为如下几点:

1. 未提交读 (READ UNCOMMITTED)

一个事务中对数据所做的修改,即便没有提交,这个修改对其他的事务仍是可见的,这种状况下就容易显现脏读,影响了数据的完全性.

举例:小明在用支付宝支付时,查看了银行卡的余额还有 300 块,其实只要 100 块,只是由于他女伴侣正在向银行卡存钱了 200 块,此时女伴侣不想存了,点击了回滚操纵,小明停止支付却失败了.

2. 读提交 (READ COMMITTED)

一个事务开端时,只能看见其他已经提交过的事务。这种状况下容易显现不成反复读 (两次读的结果不一样).

举例:一样用上面的例子举例,当他女伴侣在刷卡时卡里余额有 100 块,但是在点击终究支付时,提醒余额不足,此时看卡里的钱没了。这是由于小明女伴侣在支付时,小明操纵的事务还未提交,所以小明女伴侣两次看到的结果不一样.

3. 可反复读 (REPEATABLE READ)

屡次读取记载的结果都是一致的,可反复读可以解决上面的不成反复读的状况。但是有这样一种状况,当一个事务在读取某个范畴的记载时,别的一个事务在这个范畴内插入了一条新的数据,当事务再次停止读取数据时,发明比第一次读取记载多了一条,这就是所谓的幻读,两次读取的结果不一致.

举例:小明女伴侣在查看银行卡的记载时,看见有 5 条消耗记载,此时小明正在消耗,这时候消耗记载里面记载了这条消耗记载,当女伴侣再次读取记载时,发明有 6 笔记录了.

4. 可串行 (SERIALIZABLE)

串行就像一个队列一个样,每个事务都是排队等待着施行,只要前一个事务提交之后,下一个事务才能停止操纵。这种状况虽然可以解决上面的幻读,但是他会在每一条数据上加一个锁,容易致使大量的锁超时和锁竞争,特殊不适用在一些高并发的业务场景下.

举例:我们在银行排队存钱,只要前一个人全部操纵完,下一个人才可以停止办理。中心的人是不成以插队的,只能一个一个的排对,事务的串行就是这样的一个概念,其实所谓的串行模式都是这样的一个概念.

2e1e764d12f9079728d6ffbafd038d4.png

e7fe7f60446f376e7055338767ac817.png

隔离性总结

通过上面的举例,我们不难发明。脏读和不成反复读重在更新数据,然后幻读重在插入数据.

多种储备引擎时事务的处置方式

按照上面事务使用的前提,我们可以得知有的储备引擎是不支撑事务的,例如 MyISAM 储备引擎就不支撑。那假如在一个事务中使用了事务性的储备引擎和非事务性的储备,提交是可以正常停止,但是回滚非事务性的储备引擎则会显示响应的错误信息,详细信息和储备引擎有关.

怎样使用事务

MySQL 中事务隐式开启的,也就是说,一个 sql 语句就是一个事务,当 sql 语句施行完毕,事务就提交了。在演示的历程中,我们显式开启.

MySQL 中的主动提交

上面提到了 MySQL 中事务是隐式开启的,则代表我们每一个 sql 是主动提交的,需要关闭则需要设定 autocommit 选项.

// 查看autocommit配置值(1或者ON则表示开启)
mysql root@127.0.0.1:(none)> show variables like '%autocommit%';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit | ON |
+---------------+-------+
1 row in set
Time: 0.018s
// 设定autocommit配置值
mysql root@127.0.0.1:(none)> set autocommit = 0;
Query OK, 0 rows affected
Time: 0.000s
mysql root@127.0.0.1:(none)> show variables like '%autocommit%';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit | OFF |
+---------------+-------+
1 row in set
Time: 0.013s

1. 表构造如下

mysql root@127.0.0.1:test> desc user;
+-------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------+--------------+------+-----+---------+----------------+
| id | int(11) | NO | PRI | <null> | auto_increment |
| name | varchar(255) | YES | | <null> | |
| age | int(2) | YES | | <null> | |
+-------+--------------+------+-----+---------+----------------+
3 rows in set
Time: 0.013s

SQL 语句

CREATE TABLE `test`.`Untitled` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL,
`age` int(2) NULL DEFAULT NULL,
PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 3 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic;

2. 使用事务

MySQL 实现事务

下面的代码,我们主要做了如下几个操纵

a. 开启事务

b. 修改数据

c. 查询数据可否改动

d. 数据回滚

e. 再次查询数据,发明数据变回修改此前的状态

f. 修改数据

g. 事务提交

h. 查询数据,发明数据变为最后一次修改的状态

i. 尝试事务回滚

j. 查询验证可否被回滚了,发明数据还是为最后一次修改的状态,事务回滚失败

// 我们先查看表中的数据,id为1的age字段是12
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张三 | 12 |
| 2 | 李四 | 15 |
+----+------+-----+
2 rows in set
Time: 0.013s
// 开启事务
mysql root@127.0.0.1:test> begin;
Query OK, 0 rows affected
Time: 0.001s
// 将id为1的age字段改为10
mysql root@127.0.0.1:test> update user set age=10 where id=1;
Query OK, 1 row affected
Time: 0.001s
// 再次查询数据时,发明数据改为修改后的值
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张三 | 10 |
| 2 | 李四 | 15 |
+----+------+-----+
2 rows in set
Time: 0.012s
// 此时我们停止回滚操纵
mysql root@127.0.0.1:test> rollback;
Query OK, 0 rows affected
Time: 0.001s
// 再次查询发明数据回到最初状态
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张三 | 12 |
| 2 | 李四 | 15 |
+----+------+-----+
2 rows in set
Time: 0.019s
// 我们再次对数据停止修改
mysql root@127.0.0.1:test> update user set age=15 where id=1;
Query OK, 1 row affected
Time: 0.001s
// 此时将事务停止提交
mysql root@127.0.0.1:test> commit;
Query OK, 0 rows affected
Time: 0.000s
// 发明此时的数据变为我们终究提交的值
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张三 | 15 |
| 2 | 李四 | 15 |
+----+------+-----+
2 rows in set
Time: 0.012s
// 我们尝试用方才回滚的方式停止复原数据
mysql root@127.0.0.1:test> rollback;
Query OK, 0 rows affected
Time: 0.000s
// 发明数据没法回退了,依然是提交后的数据
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张三 | 15 |
| 2 | 李四 | 15 |
+----+------+-----+
2 rows in set
Time: 0.017s

PHP 实现事务实例代码

<?php
// 连接MySQL
$mysqli = new mysqli('127.0.0.1', 'root', '123456', 'test', 3306);
// 关闭事务主动提交
$mysqli->autocommit(false);
// 1.开启事务
$mysqli->begin_transaction();
// 2.修改数据
$mysqli->query("update user set age=10 where id=1");
// 3.查看数据
$mysqli->query("select * from user");
// 4.事务回滚
$mysqli->rollback();
// 5.查看数据
$mysqli->query("select * from user");
// 7.修改数据
$mysqli->query("update user set age=15 where id=1");
// 8.事务提交
$mysqli->commit();
// 9.事务回滚
$mysqli->rollback();
// 10.查看数据
$mysqli->query("select * from user");

怎样设定事务的隔离级别

// 查看当前的事务隔离级别
mysql root@127.0.0.1:test> select @@tx_isolation;
+-----------------+
| @@tx_isolation |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set
Time: 0.015s
// 设定隔离级别
set session transaction isolation level 隔离级别(上面事务隔离级别中的英文单词);

以上就是MySQL 事务最全详解的具体内容,更多请关注百分百源码网其它相关文章!

打赏

打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,你说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

百分百源码网 建议打赏1~10元,土豪随意,感谢您的阅读!

共有154人阅读,期待你的评论!发表评论
昵称: 网址: 验证码: 点击我更换图片
最新评论

本文标签

广告赞助

能出一分力是一分吧!

订阅获得更多模板

本文标签

广告赞助

订阅获得更多模板