您现于de位置乃:亚博 > Mysql

亚博 2019-09-11 Mysql 46

MySQL 事务特点及详解

什么乃事务?

用 MySQL 官方de1句话来描述事务乃什么?

MySQL 事务主要用于处理操作量大 复杂度高de数据.那何为数据量大?何为复杂度高呢?我用我自己de理解来描述1下吧.事务其实就乃 MySQL 中处理数据de1种方式 主要用于数据完整性高 数据之间依赖性大de情况下de1种数据处理方式.

举个例子 小张向小李de银行卡打 200 块钱 于小张点击了确认转账de按钮时 系统突然崩溃了.会出现这样几中不正确de情况:

1. 小张de钱打到小李de账户上 但乃自己de账户上de钱没被扣.

2. 小张de钱打没到小李de账户上了 但乃自己账户上de钱被扣.

这样de业务场景就需要 MySQL 事务保持 即使机器出故障de情况下 数据仍然乃正确de.

事务使用de条件

MySQL 要使用事务 需要 MySQL 中de存储引擎支持.现目前 MySQL 内置de存储引擎支持事务de有 InnoDB NDB cluster, 第3方de存储引擎有 PBXT 和 XtrDB.

事务有什么特点?

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

原子性 (atomicity):

1个事务必须被作为1个不可分割de最小工作单块 每个事务中de所有操作必须要么成功 或者要么失败 永远不可能1些操作失败 1些操作成功 这就乃所谓de原子性de概念.

1致性 (consistency):

1致性就像上面举de1个例子1样 当发生异常情况下 数据仍然乃正确de.就乃说当1个事务执行失败了 数据之间乃不会受异常de情况而影响 永远保持着他de正确性.

隔离性 (isolation):

当1个事务还未提交 每个事务之间乃相互隔离de 互补受到影响.

持久性 (durability):

当1个事务进行提交之后 发生de变化就会永远保存于数据库中.

事务de隔离级别

于谈及到 MySQL de隔离性de特点 就不得不说说隔离性de几种级别.至于为什么会涉及到这1点 可以这样简单de理解:如果同1时刻 有两个请求于执行事务de操作 并且这两个事务乃对同1条数据做操作 那么到底最终de结果乃以谁de为准呢?不同de隔离级别导致de结果不1样 因此事务de隔离级别也乃1个非常重要de点.

隔离级别分为如下几点:

1. 未提交读 (READ UNCOMMITTED)

1个事务中对数据所做de修改 即使没有提交 这个修改对其他de事务仍乃可见de 这种情况下就容易出现脏读 影响了数据de完整性.

举例:小明于用支付宝支付时 查看了银行卡de余额还有 300 块 其实只有 100 块 只乃因为他女朋友正于向银行卡存款了 200 块 此时女朋友不想存了 点击了回滚操作 小明进行支付却失败了.

2. 读提交 (READ COMMITTED)

1个事务开始时 只能看见其他已经提交过de事务.这种情况下容易出现不可重复读 (两次读de结果不1样).

举例:同样用上面de例子举例 当他女朋友于刷卡时卡里余额有 100 块 但乃于点击最终支付时 提示余额不足 此时看卡里de钱没了.这乃因为小明女朋友于支付时 小明操作de事务还未提交 所以小明女朋友两次看到de结果不1样.

3. 可重复读 (REPEATABLE READ)

多次读取记录de结果都乃1致de 可重复读可以解决上面de不可重复读de情况.但乃有这样1种情况 当1个事务于读取某个范围de记录时 另外1个事务于这个范围内插入了1条新de数据 当事务再次进行读取数据时 发现比第1次读取记录多了1条 这就乃所谓de幻读 两次读取de结果不1致.

举例:小明女朋友于查看银行卡de记录时 看见有 5 条消费记录 此时小明正于消费 这时候消费记录里面记录了这条消费记录 当女朋友再次读取记录时 发现有 6 条记录了.

4. 可串行 (SERIALIZABLE)

串行就像1个队列1个样 每个事务都乃排队等候着执行 只有前1个事务提交之后 下1个事务才能进行操作.这种情况虽然可以解决上面de幻读 但乃他会于每1条数据上加1个锁 容易导致大量de锁超时和锁竞争 特别不适用于1些高并发de业务场景下.

举例:我们于银行排队存钱 只有前1个人全部操作完 下1个人才可以进行办理.中间de4瞬豢梢圆宥觗e 只能1个1个de排对 事务de串行就乃这样de1个概念 其实所谓de串行模式都乃这样de1个概念.

隔离性总结

通过上面de举例 我们不难发现.脏读和不可重复读重于更新数据 然后幻读重于插入数据.

多种存储引擎时事务de处理方式

根据上面事务使用de条件 我们可以得知有de存储引擎乃不支持事务de 例如 MyISAM 存储引擎就不支持.那如果于1个事务中使用了事务性de存储引擎和非事务性de存储 提交乃可以正常进行 但乃回滚非事务性de存储引擎则会显示响应de错误信息 具体信息和存储引擎有关.

如何使用事务

MySQL 中事务隐式开启de 也就乃说 1个 sql 语句就乃1个事务 当 sql 语句执行完毕 事务就提交了.于演示de过程中 我们显式开启.

MySQL 中de自动提交

上面提到了 MySQL 中事务乃隐式开启de 则代表我们每1个 sql 乃自动提交de 需要关闭则需要设置 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 实现事务

下面de代码 我们主要做了如下几个操作

a. 开启事务

b. 修改数据

c. 查询数据乃否改变

d. 数据回滚

e. 再次查询数据 发现数据变回修改之前de状态

f. 修改数据

g. 事务提交

h. 查询数据 发现数据变为最后1次修改de状态

i. 尝试事务回滚

j. 查询验证乃否被回滚了 发现数据还乃为最后1次修改de状态 事务回滚失败

// 我们先查看表中de数据,id为1deage字段乃12
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张3 | 12 |
| 2 | 李4 | 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为1deage字段改为10
mysql root@127.0.0.1:test> update user set age=10 where id=1;
Query OK, 1 row affected
Time: 0.001s
// 再次查询数据时,发现数据改为修改后de值
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张3 | 10 |
| 2 | 李4 | 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 | 张3 | 12 |
| 2 | 李4 | 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
// 发现此时de数据变为我们最终提交de值
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张3 | 15 |
| 2 | 李4 | 15 |
+----+------+-----+
2 rows in set
Time: 0.012s
// 我们尝试用刚才回滚de方式进行还原数据
mysql root@127.0.0.1:test> rollback;
Query OK, 0 rows affected
Time: 0.000s
// 发现数据无法回退了,仍然乃提交后de数据
mysql root@127.0.0.1:test> select * from user;
+----+------+-----+
| id | name | age |
+----+------+-----+
| 1 | 张3 | 15 |
| 2 | 李4 | 15 |
+----+------+-----+
2 rows in set
Time: 0.017s


APP 实现事务实例代码

<?app
// 连接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");

如何设置事务de隔离级别

// 查看当前de事务隔离级别
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 隔离级别(上面事务隔离级别中de英文单词);

以上就乃MySQL 事务最全详解de详细内容 更多请关注app中文网其它相关文章!

上1篇:没有了

下1篇:Navicat Premium12远程连接MySQL数据库

评论