浅谈mysql事务
数据库是一切工作的基础,没有数据其他工作也将失去意义-----不知道哪位大牛说的,也可能是我瞎掰的(嗨嗨)
一.理解ACID(原子性、一致性、隔离性、持久性)并不是专属于mysql数据库
来一个例子吧:
银行数据库中有两张表:信用卡表(checking),储蓄表(savings)。现在要从用户A的信用卡账户转移1000块到他的储蓄账户之中,那么最起码需要以下步骤:
1.检查信用卡账户的余额大于1000块
2.将信用卡账户的余额减去1000块
3.将储蓄账号的余额加上1000块
上面三个步骤必须打包在一个事务之中,要失败就全部回滚三个步骤
事务sql的样本:
(1)start transaction;
(2)select balance from checking where customer_id='1110000'
(3)update checking set balance=balance-1000 where customer_id='1110000'
(4)update savings set balance=balance+1000 where customer_id='1110000'
(5)commit;
我们想一个问题:
如果上述语句执行到第四句的时候服务器崩溃了,会出现什么情况?
是用户损失了1000块,还是银行损失了1000块,还是什么其他的状况?天知道
现在再去理解的ACID的概念是不是很简单了,当然,ACID在现实中要完全实现非常困难,甚至不可能。真正的实现也没有这里说的如此简单,这里只能阐述一下概念。
1.原子性保证一个事务为一个最小的单元,内部不可分割;(可以理解为最小执行单元)。整个事务中的所有操作要么全部提交成功,要么全部失败回滚,不可能执行其中的一部分。
2.数据库总是从一个一致性的状态转换到另一个一致性的状态。在这个例子中,一致性可以确保即使数据库崩溃了(没有commit之前),信用卡账户也不会损失1000块。因为还没有提交,事务所做的修改也不会保存到数据库中去。
3.隔离性保证不同事务间看到的数据视图相互独立,相互隔离(隔离级别可设置),以后会讨论一下关于隔离级别(Isolation level)的问题;
4.持久性保证事务提交后数据会持久的保存下来;
二.使用事务
MySQL的事务支持不是绑定在MySQL服务器本身,而是与存储引擎相关1.MyISAM:不支持事务,用于只读程序提高性能 2.InnoDB:支持ACID事务、行级锁、并发 3.Berkeley DB:支持事务(我抄的,嗨嗨!!)
(1)提交事务(关闭事务)
COMMIT:提交
(2) 放弃事务(关闭事务)
ROLLBACK:回滚
注:设立回滚点savepoint
SAVEPOINT point1
ROLLBACK TO SAVEPOINT point1
发生在回滚点 point1之前的事务被提交,之后的被忽略
(3)MYSQL的事务处理的两种方法
1.用begin,rollback,commit来实现
begin开始一个事务
rollback事务回滚
commit 事务确认(如果失去数据库连接则没有commit的事务都会失去)
2.直接用set来改变mysql的自动提交模式
mysql默认是自动提交的,也就是你提交一个query,就直接执行!可以通过
set autocommit = 0 禁止自动提交
set autocommit = 1 开启自动提交
注:当用set autocommit = 0 的时候,你以后所有的sql都将作为事务处理,直到你用commit确认或 rollback结束,注意当你结束这个事务的同时也开启了新的事务!按第一种方法只将当前的做为一个事务!
三.事务的锁定(此段摘自http://www.cnblogs.com/ymy124/p/3718439.html,感谢:网名还没想好)
mysql系统默认: 不需要等待某事务结束,可以直接查询结果,但是无法进行修改、删除。
缺点:可能查询的是过期的结果。
优点:可直接查询结果,不需要等待事务结束。
分类:
1、SELECT …… LOCK IN SHARE MODE(共享锁)
查询到的数据,就是数据库在这一时刻的数据(其他已commit事务的结果,已经反应到这里了)
SELECT 必须等待,某个事务结束后才能执行
2、SELECT …… FOR UPDATE(排它锁)
例如 SELECT * FROM tablename WHERE id<200
那么id<200的数据,被查询到的数据,都将不能再进行修改、删除、SELECT …… LOCK IN SHARE MODE操作
一直到此事务结束
共享锁 和 排它锁 的区别:在于是否阻断其他客户发出的 SELECT …… LOCK IN SHARE MODE命令
3、INSERT / UPDATE / DELETE
所有关联数据都会被锁定,加上排它锁
4、防插入锁
例如 SELECT * FROM tablename WHERE id>200
那么id>200的记录无法被插入
5、死锁
自动识别死锁
先进来的进程被执行,后来的进程收到出错消息,并按ROLLBACK方式回滚
innodb_lock_wait_timeout = n 来设置最长等待时间,默认是50秒
四.事务的隔离(转自:http://xm-king.iteye.com/blog/770721)
SQL标准定义了4类隔离级别,包括了一些具体规则,用来限定事务内外的哪些改变是可见的,哪些是不可见的。低级别的隔离级一般支持更高的并发处理,并拥有更低的系统开销。
1.Read Uncommitted(读取未提交数据): 在这个隔离级别,所有事务都可以看到其他未提交事务的执行结果,脏读(Dirty Read)实际用的并不多。
2.Read committed(读取提交):这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)。它满足了隔离的简单定义:一个事务只能看见已经提交事务所做的改变。这种隔离级别 也支持所谓的不可重复读(Nonrepeatable Read),因为同一事务的其他实例在该实例处理其间可能会有新的commit,所以同一select可能返回不同结果。
3.Repeatable Read(可重复读取):这是MySQL的默认事务隔离级别,它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行。不过理论上,这会导致另一个棘手的问题:幻读 (Phantom Read)。简单的说,幻读指当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,当用户再读取该范围的数据行时,会发现有新的“幻影” 行。InnoDB和Falcon存储引擎通过多版本并发控制(MVCC,Multiversion Concurrency Control)机制解决了该问题。
4.Serializable(可序列化):这是最高的隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。简言之,它是在每个读的数据行上加上共享锁。在这个级别,可能导致大量的超时现象和锁竞争。