数据库

事务

原子性(Atomicity)

原子性:记录之前的版本,允许回滚

一致性(Consistency)

一致性:事务开始和结束之间的中间状态不会被其他事务看到

持久性(Durability)

一个事务一旦提交,他对数据库的修改应该永久保存在数据库中

隔离性(Isolation)

问题:如果两个并发事务(A事务和B事务)修改数据库表同一行数据,那会出现什么情况。
可能最后结果是以A事务覆盖B事务结果,或者反过来。
这时候数据库需要有个机制进行控制,避免这种情况出现,那就是隔离级别,数据库提供了多种隔离级别。

隔离性是当多个用户并发访问数据库时,比如操作同一张表时,数据库为每一个用户开启的事务,不能被其他事务的操作所干扰,多个并发事务之间要相互隔离
即要达到这么一种效果:对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这不变成T1,T2串行了嘛
我认为这个只是局部的串行,也就是只有在相互影响的事务之间,因为设置隔离级别后,会对影响的行或者表进行加锁,不会对整个数据库的表或者行加锁。所以不会影响其它事务的并行执行。

事务隔离级别

前面说的事务的四大特性,都只是理论上的。实际上的控制是由事务隔离级别来做的。
事务的隔离级别需要解决的问题

  • 脏读
    某个事务读取到了另一个事务未提交的数据。
    但是要区分一种情况,
    当前事务提交的数据自己会读到,这不算脏读
  • 不可重复读
    事务在执行过程中,前后两次读取到的数据不一致。这中间肯定发生了其它事务提交或者修改了数据,按理说这也没错,毕竟其它事务修改了数据并提交了,你读到也是正常的。但是从程序控制来说,两次读到不同数据,会对程序逻辑产生无可估计的影响以及错误。
    描述:事务T1在读取某一数据,而事务T2立马修改了这个数据并且提交事务给数据库,但此时事务T1还没执行完,事务T1再次读取该数据就得到了不同的结果,发送了不可重复读。
  • 虚读,幻读
    事务T1对一个表中所有的行的某个数据项做了从“1”修改为“2”的操作,这时事务T2又对这个表中插入了一行数据项,而这个数据项的数值还是为“1”并且提交给数据库。而操作事务T1的用户如果再查看刚刚修改的数据,会发现还有一行没有修改,其实这行是从事务T2中添加的,就好像产生幻觉一样,这就是发生了幻读。

幻读和不可重复读都是读取了另一条已经提交的事务(这点就脏读不同),所不同的是不可重复读查询的都是同一个数据项,而幻读针对的是一批数据整体(比如数据的个数)。

隔离级别

【1】mysql事务隔离级别
https://www.cnblogs.com/dooor/p/5303904.html
【2】java事务及隔离级别http://blog.csdn.net/wang379275614/article/details/24818397

  • Serializable (串行化):可避免脏读、不可重复读、幻读的发生
  • Repeatable read (可重复读):可避免脏读、不可重复读的发生
  • Read committed (读已提交):可避免脏读的发生
  • Read uncommitted (读未提交):最低级别,任何情况都无法保证

隔离级别的设置

隔离级别设置是会话级别的,数据库会有个默认或者全局的会话级别设置。

-- MYSQL
SET [SESSION|GLOBAL] TRANSACTION ISOLATION LEVEL 
[
READ UNCOMMITTED
|READ COMMITTED
|REPEATABLE READ
|SERIALIZABLE
]

事务隔离级别与数据库锁的关系

事务隔离级别是通过加锁的机制实现的。
隔离级别的基础是锁和版本号。

java多数据源

JTA
XA连接不支持JDBC的自动提交特性, 也就是说应用程序不必在XA连接上调用java.sql.Connection.commit()或java.sql.Connection.rollback()。相反,应用程序应该使用UserTransaction.begin()、UserTransaction.commit()和UserTransaction.rollback()
两阶段提交

资料

【1】java和mysql事务
https://www.cnblogs.com/fjdingsd/p/5273008.html

你可能感兴趣的:(数据库)