事务可以保证多个操作原子性,要么全成功,要么全失败。对于数据库来说事务保证批量的DML要么全成功,要么全失败。事务具有四个特征ACID
原子性(Atomicity)
整个事务中的所有操作,必须作为一个单元全部完成(或全部取消)。
一致性(Consistency)
在事务开始之前与结束之后,数据库都保持一致状态。
隔离性(Isolation)
一个事务不会影响其他事务的运行。
持久性(Durability)
在事务完成以后,该事务对数据库所作的更改将持久地保存在数据库之中,并不会被回滚。
事务中存在一些概念:
当执行DML语句是其实就是开启一个事务
关于事务的回滚需要注意:只能回滚insert、delete和update语句,不能回滚select(回滚select没有任何意义),对于create、drop、alter这些无法回滚.
事务只对DML有效果。
注意:rollback,或者commit后事务就结束了。
1、创建表
mysql> create table user(
-> id int(11) primary key not null auto_increment,
-> username varchar(30),
-> password varchar(30)
-> ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
Query OK, 0 rows affected (0.13 sec)
2、查询表中数据
mysql> select * from user;
Empty set
3、开启事务
mysql> START TRANSACTION;
Query OK, 0 rows affected (0.15 sec)
4、插入数据
mysql> insert into user(username,password) values('zhangsan','123');
Query OK, 1 row affected (0.19 sec)
5、查看数据
mysql> select * from user;
+----+----------+----------+
| id | username | password |
+----+----------+----------+
| 1 | zhangsan | 123 |
+----+----------+----------+
1 row in set (0.32 sec)
6、修改数据,并查看修改后的数据
mysql> update user set username='lisi' where id=1;
Query OK, 1 row affected (0.26 sec)
Rows matched: 1 Changed: 1 Warnings: 0
mysql> select * from user;
+----+----------+----------+
| id | username | password |
+----+----------+----------+
| 1 | lisi | 123 |
+----+----------+----------+
1 row in set (0.35 sec)
7、回滚事务,并查看user表
mysql> rollback;
Query OK, 0 rows affected (0.26 sec)
mysql> select * from user;
Empty set
自动提交模式用于决定新事务如何及何时启动。
启用自动提交模式:
禁用自动提交模式:
自动提交模式可以通过服务器变量AUTOCOMMIT来控制。
例如:
mysql> SET AUTOCOMMIT = OFF;
mysql> SET AUTOCOMMIT = ON;
或
mysql> SET SESSION AUTOCOMMIT = OFF;
mysql> SET SESSION AUTOCOMMIT = ON;
show variables like '%auto%'; -- 查看变量状态
事务的隔离级别决定了事务之间可见的级别。
当多个客户端并发地访问同一个表时,可能出现下面的一致性问题:
一个事务开始读取了某行数据,但是另外一个事务已经更新了此数据但没有能够及时提交,这就出现了脏读取。
在同一个事务中,同一个读操作对同一个数据的前后两次读取产生了不同的结果,这就是不可重复读。
幻像读是指在同一个事务中以前没有的行,由于其他事务的提交而出现的新行。
InnoDB 实现了四个隔离级别,用以控制事务所做的修改,并将修改通告至其它并发的事务:
允许一个事务可以看到其他事务未提交的修改。
允许一个事务只能看到其他事务已经提交的修改,未提交的修改是不可见的。
确保如果在一个事务中执行两次相同的SELECT语句,都能得到相同的结果,不管其他事务是否提交这些修改。 (银行总账)该隔离级别为InnoDB的缺省设置。
将一个事务与其他事务完全地隔离。
例:A可以开启事物,B也可以开启事物
A在事物中执行DML语句时,未提交
B不以执行DML,DQL语句
隔离级别 | 脏读取 | 不可重复读 | 幻象读 |
---|---|---|---|
读未提交 | 可能 | 可能 | 可能 |
读已提交 | 不可能 | 可能 | 可能 |
可重复读 | 不可能 | 不可能 | 对InnoDB不可能 |
串行化 | 不可能 | 不可能 | 不可能 |
可以在my.ini
文件中使用transaction-isolation
选项来设置服务器的缺省事务隔离级别
该选项值可以是:
例如:
[mysqld]
transaction-isolation = READ-COMMITTED
隔离级别也可以在运行的服务器中动态设置,应使用SET TRANSACTION ISOLATION LEVEL语句。
其语法模式为:
SET [GLOBAL | SESSION] TRANSACTION ISOLATION LEVEL
其中的可以是:
例如: SET TRANSACTION ISOLATION LEVEL **REPEATABLE READ**;
事务隔离级别的作用范围分为两种:
例如,设置会话级隔离级别为READ COMMITTED
mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
或:
mysql> SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
设置全局级隔离级别为READ COMMITTED :
mysql> SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;
服务器变量tx_isolation(包括会话级和全局级两个变量)中保存着当前的会话隔离级别。
为了查看当前隔离级别,可访问tx_isolation变量:
查看会话级的当前隔离级别:
mysql> SELECT @@tx_isolation;
或:
mysql> SELECT @@session.tx_isolation;
//查看全局级的当前隔离级别:
mysql> SELECT @@global.tx_isolation;
会话一 | 会话二 |
---|---|
mysql> prompt s1> | mysql> use bjpowernode |
s1>use bjpowernode | mysql> prompt s2> |
s1>create table tx (id int(11),num int (10)); | |
s1>set global transaction isolation level read uncommitted; | |
s1>start transaction; | |
s2>start transaction; | |
s1>insert into tx values (1,10); | |
s2>select * from tx; | |
s1>rollback; | |
s2>select * from tx; |
会话一 | 会话二 |
---|---|
s1> set global transaction isolation level read committed; | |
s1>start transaction; | |
s2>start transaction; | |
s1>insert into tx values (1,10); | |
s1>select * from tx; | |
s2>select * from tx; | |
s1>commit; | |
s2>select * from tx; |
会话一 | 会话二 |
---|---|
s1> set global transaction isolation level repeatable read; | |
s1>start transaction; | s2>start transaction; |
s1>select * from tx; | |
s1>insert into tx values (1,10); | |
s2>select * from tx; | |
s1>commit; | |
s2>select * from tx; |