您的位置:首页 > 数据库

事务了解

2016-07-19 15:24 169 查看
附录: 事务的ACID(Atomicity \Consistency \Isolation \Durablility)

A: 事务必须是原子(不可分割),要么执行成功进入下一个状态,要么失败rollback 到最初状态。

C:在事务开始之前和事务结束以后,数据库的完整性约束没有被破坏。 这个一般通过外键来约束。

I:一个事务不能知道另外一个事务的执行情况(中间状态)

D:在事务完成以后,该事务所对数据库所作的更改便持久的保存在数据库之中,并不会被回滚。

mysql 自己的MyISAM 没有通过acid 测试,但是InnoDB 可以做到。

在分布式的系统中,通常会有多个线程连接到数据库中同时对一个表进行操作(这里的同时并不表示同一个时间点,而是同时竞争cpu的资源,至于如何调度,就要看线程和操作系统如何进行调度了),这种情况下如果会话的事物设置不当,就会导致数据混乱,常常会出现以下三种情况(假设现在系统中有两个会话A和B,同时对表T_Test操作):

1.脏读:如果有A向B 做了这个操作:update account set money=money+100 where name=’B’;在没有commit 之前B 查询:select money from account where name=’B’;找到了没有提交的money ,之后A在此时有rollback ,B 再查询,100 不见了。为了避免提高级别:read committed 。就是只能读取提交后的东东。

2.不可重复读:1中说明的就是我们不能读取一个事务的中间状态。 而重复读是指我们每次读取到的结果都要一直。 这个也是mysql 默认的级别。

mysql> select @@tx_isolation ;

+—————–+

| @@tx_isolation |

+—————–+

| REPEATABLE-READ |

+—————–+

3.幻读:在一个事务内读取到了别的事务插入的数据,导致前后读取不一致。和不可重复读的区别是:不可重复读是读取到了别人对表中的某一条记录进行了修改,导致前后读取的数据不一致。 虚读是前后读取到表中的记录总数不一样,读取到了其它事务插入的数据。比如现在有 A 和 B 两个应用程序,他们并发访问了数据库中的某一张表,假设表中有 3 条记录,B 执行查询操作, 第一次查询表得到了 3 条记录。此时 A 对表进行了修改,增加了一条记录,当 B 再次查询表的时候,发现多了一条数据。这种情况就造成了 B 的虚读。但是虚读是不一定每次都发生的,这种情况是不确定的。为了避免虚读,我们可以将事物隔离级别设置为 serializable 如果设置成了这种级别,那么数据库就变成了单线程访问的数据库,导致性能降低很多。

summary:

(1)Serializable:可避免脏读、不可重复读、虚读情况的发生。

(2)Repeatable read:可避免脏读、不可重复读情况的发生。(可重复读,是 mysql 默认的事务隔离级别)

(3)Read committed:可避免脏读情况发生。(读取已提交的数据)

(4)Read uncommitted:最低级别,以上情况均无法保证。(读取到了未提交的数据)

当我们将数据库的隔离级别设置为:Serializable 的时候,虽然可以避免所有并发访问的问题,但是 Serializable 采用的是单线程来解决并发访问的问题,也就是说在某一段时间内,只能有一个用户对数据库进行操作,导致其它用户阻塞。导致数据库的访问性能很差。

1.读未提交(Read Uncommitted):这种隔离级别可以让当前事务读取到其它事物还没有提交的数据。这种读取应该是在回滚段中完成的。通过上面的分析,这种隔离级别是最低的,会导致引发脏读,不可重复读,和幻读。

2.读已提交(Read Committed):这种隔离级别可以让当前事务读取到其它事物已经提交的数据。通过上面的分析,这种隔离级别会导致引发不可重复读,和幻读。

3.可重复读取(Repeatable Read):这种隔离级别可以保证在一个事物中多次读取特定记录的时候都是一样的。通过上面的分析,这种隔离级别会导致引发幻读。

4.串行(Serializable):这种隔离级别将事物放在一个队列中,每个事物开始之后,别的事物被挂起。同一个时间点只能有一个事物能操作数据库对象。这种隔离级别对于数据的完整性是最高的,但是同时大大降低了系统的可并发性。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  数据库