事务是区分文件存储系统与Nosql数据库重要特性之一,其存在的意义就是为了保证正确执行crud操作,如何才算正确操作?于是引出了事务需要保证4个特性ACID。
事务的基本特性
- 原子性(atomicity):事务各项操作,要么全部成功要么全部失败,及任何一个操作失败,整个事务就会失败。
- 一致性(consistency):事务结束后系统状态是一致的。(例如:银行转账,转账最终总是是一致的)
- 隔离性(islatioin):并发执行彼此事务无法看到对方状态。
- 持久性(isolation):事务完成后数据将会被持久化,即使发生灾难性的失败。
在高并发情况下,要完全保证ACID是非常困难的,除非把所有的事务操作串行化执行,但是性能将会大大降低,所有数据库设计了4中隔离级别,提供给用户争对不同中情况,基于业务作出选择:
隔离级别 | 脏读 (Dirty Read) | 不可重复读(NonRepeatable Read) | 幻读(Phantom Read) |
---|---|---|---|
未提交读(Read uncommitted) | 可能 | 可能 | 可能 |
已提交读(Read Committed) | 不可能 | 可能 | 可能 |
可重复读(Repeatable) | 不可能 | 不可能 | 可能 |
串行化(Serializable) | 不可能 | 不可能 | 不可能 |
- 脏读:一个事务读取到另外一个事务未提交的更新数据
- 不可重复读:在同一个事务中,多次读取通一数据返回的结果有所不同,后续读取可以读取到另一事务已提交的更新数据,相反,“可重复读”在同一事物中多次读取数据时,能够保证所读取到的数据一致,也就是后续读取不能读取到另一个事务已提交的更新数据。
- 幻读:查询表中一条数据如果不存在就插入一条,并发时候确发现,数据库里存在两条一样的数据,这就是幻读问题(一般使用幂等性保证)
mysql数据库,默认隔离级别为:可重复读(Repeatable)
事务传播机制
类别 | 事务传播类型 | 说明 |
支持当前事务 | PROPAGATION_REQUIRED(必须的) | 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,就加入到这个事务中,这是最常见的选择 |
PROPAGATION_SUPPORTS(支持) | 支持当前事务,如果当前没有事务,就以非事务方式执行 | |
PROPAGATION_MANDATORY(强制) | 使用当前事务,如果当前没有事务,就抛出异常 | |
不支持当前事务 | PROPAGATION_REQUIRED_NEW(隔离) | 新建事务,如果当前事务不存在,就把当前事务挂起 |
PROPAGATION_NOT_SUPPORTS(不支持) | 以非事务方式执行,如果当前事务存在,就把当前事务挂起 | |
PROPAGATION_NEVER(强制非事务) | 以非事务方式执行,如果当前存在事务就抛出异常 | |
嵌套事务 | PROPAGATION_NESRWD(嵌套事务) | 如果当前存在事务,则在嵌套事务内执行,如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作 |
- 挂起:产生一个新的连接,建一个新的事务
- 什么操作属于同一事务?
新建连接
conn = new Connention()
conn.setAutoCommit(false)
// 增,删,改,查
conn.commit 或者 conn.rollback 或 conn.close()
之间都属于同一事物