什么是事务?
数据库事务(database transaction),是指作为单个逻辑工作单元执行的一系列操作,要么完全的执行,要么完全不执行。
简单的说:事务就是将一堆的SQL语句(通常是增删改查)绑定在一起执行,要么都执行成功,要么都执行失败,即都执行成功才算成功,否则就会恢复到这堆SQL执行之前的状态。
事务的四个特性ACID
原子性(Atomicity,或称不可分割性)
一个事务中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。事务在执行过程中如果发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务没有执行过一样
一致性(Consistency)
在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的资料必须完全符合所有的预设规则,包含资料的精确度、串联性以及后续数据库可以自发性的完成预定的工作。
隔离性(Isolation,又称独立性)
同一时间,只允许一个事务请求同一数据,不同的事务之间彼此没有任何干扰。数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致
持久性(Durability)
事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失
隔离级别
读未提交(Read uncommitted) 安全性最差,可能发生并发数据问题,性能最好
读提交(Read committed)Oracle默认的隔离级别
可重复读(repeatable read)MySQL默认的隔离级别,安全性较好,性能一般
串行化(Serializable) 表级锁,读写都加锁,效率低下,安全性能高,不能并发
事务的并发问题:
脏读:(读取未提交数据)指一个事务读取了另一个事务没有提交的数据
解决:隔离级别设置为读提交
不可重复读:(前后多次读取,数据内容不一致) 在一个事务读取表中的某一行数据,多次读取结果不同(不一定是错误,只是某些场合不对),针对update操作
解决:使用行级锁,锁定该行,事务A多次读取完成后才释放该锁,这个时候才允许其他事务更改刚才的数据(隔离级别设置为可重复读)
幻读(虚读):(前后多次读取,数据总量不一致)是指在一个事务读取到了别的事务插入的数据,导致前后读取不一致(一般是行影响,多了一行),针对insert和delete操作
解决:使用表级锁,锁定整张表,事务A多次读取数据总量后才释放该锁,这个时候才允许其他事务新增数据(隔离级别设置为序列化/
串行化)
事务隔离级别 | 脏读 | 不可重复读 | 幻读 |
读未提交(read-uncommitted) | 是 | 是 | 是 |
读提交(read-committed) | 否 | 是 | 是 |
可重复读(repeatable read) | 否 | 否 | 是 |
序列化(串行化)(Serializable) | 否 | 否 | 否 |
SET autocommit=0 -- 关闭
SET autocommit=1 -- 开启(默认)
-- 手动处理事务
-- 事务开启
START TRANSACTION -- 标记一个事务的开始,从这以后的sql都在同一个事务内
COMMIT -- 提交:持久化(成功!)
ROLLBACK -- 回滚:回到原来的样子(失败!)
-- 事务结束
SET autocommit=1 -- 开启自动提交
-- 了解
SAVEPOINT 保存点名 -- 设置一个事务的保存点
ROLLBACK SAVEPOINT 保存点名 -- 回滚到保存点
RELEASE SAVEPOINT 保存点名 -- 撤销保存点
测试事务实现转账
CREATE DATABASE `shop` CHARACTER SET utf8 COLLATE utf8_general_ci
USE shop
CREATE TABLE `account`(
`id` INT(3) NOT NULL AUTO_INCREMENT,
`name` VARCHAR(50) NOT NULL,
`money` DECIMAL(9,2) NOT NULL,PRIMARY KEY(`id`)
)ENGINE=INNODB DEFAULT CHARSET=utf8
INSERT INTO `account` (`name`,`money`)
VALUES ('小可爱',1314520.00),('小美女',5201314.00)
SET autocommit=0;-- 关闭自动提交
START TRANSACTION -- 开启一个事务
UPDATE account SET money=money-222 WHERE `name`='小可爱'
UPDATE account SET money=money+222 WHERE `name`='小美女'
COMMIT;-- 提交事务
ROLLBACK;-- 回滚
SET autocommit=1; -- 恢复默认值