一:并发事务导致的问题
1.当同一个应用程序或者不同应用程序的多个事务在同一个数据集并发执行时,可能会出现很多意外问题
二:并发事务所导致的问题,大概可以分为三个类型
1.脏读:对于两个事务T1,T2。T1读取了T2已经更新但没有提交的字段,若T2回滚,T1读取的内容就是临时且无效的
2.不可重复读:对于两个事务T1,T2。T1读取了一个字段,然后T2更新了该字段,T1再次读取同一字段,值就不同了
3.幻读:对于两个事物T1,T2,T1从一个表中读取一个字段,然后T2在该表插入一些新的行,之后,如果T1再次读取同一个表就会多出几行.
package com.dhx.spring.tx;
import java.awt.datatransfer.FlavorListener;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService {
@Autowired
private BookShopDao bookShopDao;
//事物注解
//1.propagation指定事物的传播行为,即当前的事务方法被另外一个事务方法调用时
//如何使用事务,默认取值为REQUIRED,即使用调用方法的事务
//REQUIRES_NEW:事务自己的事务,调用事务方法的事务被挂起
//2.使用isolation指定事务的隔离级别,最常用的是READ_COMMITTED
//3.默认情况下Spring的声明式事务对所有运行时的异常进行回滚,也可以通过对应属性设置
//通常情况下取默认值即可
/*@Transactional(propagation=Propagation.REQUIRES_NEW,
isolation=Isolation.READ_COMMITTED,
noRollbackFor= {UserAccountReception.class})*/
//4.使用readOnly指定事务是否为可读,表示这个事务只读取数据但不更新数据
//这样可以帮助数据库引擎优化事务,若真的是一个只读取数据库值的方法,应设置为readOnly=true
//5.timeout指定强制回滚之前事务可以占用的时间 单位是秒
@Transactional(propagation=Propagation.REQUIRES_NEW,
isolation=Isolation.READ_COMMITTED,
readOnly=false,
timeout=1)
@Override
public void purchase(String username, String isbn) {
//获得书的单价
int price=bookShopDao.findBookPriceByIsbn(isbn);
//书的库存减一
bookShopDao.updateBookStock(isbn);
//用户余额减去price
bookShopDao.updateUserAccount(username, price);
}
}