1、事务(https://tech.souyunku.com/zhai1997/p/11710082.html)
(1)事务的特性:acdi
(2)事务的并发问题:丢失修改,脏读,不可重复读
(3)事务的隔离级别:1、2、4、8
2、Spring的事务管理
(1)Spring封装了事务管理的代码:打开事务、提交事务、回滚事务
在我们学习的不同阶段(JDBC、Hibernate),对事物处理的方法是不一样的,为了解决这个问题,Spring提供了一个接口,PlatformTransactionManager(平台事务管理器),
该接口可以根据不同的平台提供不同的方法来处理事务,
(2)Spring管理事务的属性
事务的隔离级别:1:读未提交、2:读已提交、4:可重复读、8:串行化
本次事务是否只读:true:只读
事务的传播行为:
PROPAGATION REQUIRED 支持当前事务,如果不存在就新建一个(默认)
PROPAGATION_SUPPORTS 支持当前事务,如果不存在,就不使用事务
PROPAGATION MANDATORY 支持当前事务,如果不存在,抛出异常
PROPAGATION_REQUIRES_NEW 如果有事务存在,挂起当前事务,创建一个新的事务
PROPAGATION_NOT_SUPPORTED 以非事务方式运行,如果有事务存在,挂起当前事务
PROPAGATION_NEUER 以非事务方式运行,如果有事务存在,抛出异常
PROPAGATION_NESTED 如果当前事务存在,则嵌套事务执行
如果,事务method1调用事务method2,如果,method1没有开启事务,则method1需要先开启一个事务,method2也调用该事务,如果,method1,已经开启了一个事务,则method2直接用这个事务即可。
3、Spring管理事务的方式:编码式
(1)配置文件:
db.properties:
jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.jdbcUrl=jdbc:mysql:///bank_transfer
jdbc.user=root
jdbc.password=root
applicationContext.xml:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans" xmlns:util="http://www.springframework.org/schema/util"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--指定要读取的配置文件的位置-->
<context:property-placeholder location="classpath:db.properties"></context:property-placeholder>
<!--将连接池放入Spring容器-->
<bean name="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"></property>
<property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
<property name="user" value="${jdbc.user}"></property>
<property name="password" value="${jdbc.password}"></property>
</bean>
<!--Dao-->
<bean name="accountDao" class="pers.zhb.dao.AccountDaoImp">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--Service-->
<bean name="accountserviceimp" class="pers.zhb.service.AccountServiceImp">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!--核心事务管理器,依赖于连接池-->
<bean name="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--事务模板对象-->
<bean name="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager" ref="transactionManager"></property>
</bean>
</beans>
要确定好每个类之间的依赖关系。
(2) Dao层:
接口:
public interface AccountDao {
void increaseMoney(Integer id,Double money);
void decreaseMoney(Integer id,Double money);
}
实现类:
public class AccountDaoImp extends JdbcDaoSupport implements AccountDao{
@Override
public void increaseMoney(Integer id, Double money) {
String sql="update transfer set money=money+? where id = ?";
super.getJdbcTemplate().update(sql,money,id);
}
@Override
public void decreaseMoney(Integer id, Double money) {
String sql="update transfer set money=money-? where id = ?";
super.getJdbcTemplate().update(sql,money,id);
}
}
(3)Service层:
接口:
public interface AccountService {
void transfer(Integer from, Integer to,Double money);
}
实现类:
public class AccountServiceImp implements AccountService {
private AccountDao accountDao;
private TransactionTemplate transactionTemplate;
public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
this.transactionTemplate = transactionTemplate;
}
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
@Override
public void transfer(final Integer from, final Integer to,final Double money) {
transactionTemplate.execute(new TransactionCallbackWithoutResult() {
@Override
protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
accountDao.decreaseMoney(from,money);
int i=1/0;
accountDao.increaseMoney(to,money);
}
});
}
}
(4)测试类:
public static void main(String [] args){
ApplicationContext applicationContext=new ClassPathXmlApplicationContext("applicationContext.xml");//创建容器对象
AccountServiceImp accountServiceImp =(AccountServiceImp)applicationContext.getBean("accountserviceimp");
accountServiceImp.transfer(1,2,12d);
}
在service层的方法中,故意制造了错误,再发生异常后未出现转账方钱减少而收款方前未增加的情况,即:钱的总数不会变。
4、Spring事务管理方式:xml配置aop事务
(1)导入约束:
导入tx、aop、context约束。
tx:配置事务通知
aop:配置aop
context:注解
(2)导包:
(3)Dao层的接口和实现类。
(4)Service层的接口和实现类,改层调用Dao层的两个转账方法。
(5)配置文件:
jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.jdbcUrl=jdbc:mysql:///bank_transfer
jdbc.user=root
jdbc.password=root
该配置文件加jdbc前缀的目的是,与其他的功能的配置文件加以区别。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.2.xsd ">
<!--指定要读取的配置文件的位置-->
<context:property-placeholder location="classpath:db.properties"></context:property-placeholder>
<!--将连接池放入Spring容器-->
<bean name="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"></property>
<property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
<property name="user" value="${jdbc.user}"></property>
<property name="password" value="${jdbc.password}"></property>
</bean>
<!--Dao-->
<bean name="accountDao" class="pers.zhb.dao.AccountDaoImp">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--Service-->
<bean name="accountserviceimp" class="pers.zhb.service.AccountServiceImp">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!--核心事务管理器,依赖于连接池-->
<bean name="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--事务模板对象-->
<bean name="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager" ref="transactionManager"></property>
</bean>
<!--配置事务通知-->
<tx:advice id="transactionInterceptor" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="transfer" isolation="REPEATABLE_READ" propagation="REQUIRED" read-only="false"/>
</tx:attributes>
</tx:advice>
</beans>
指出数据库相关的配置文件的位置。
读取配置文件中的数据,连接池被放入到了Spring容器。
将Dao层和Service层的对象放入到Spring容器中,其中Service层依赖于Dao层。
核心事务管理器,依赖于连接池。
事务模板对象,依赖于核心事务管理器。
配置事务通知:以方法为单位,isolation:隔离级别,propagation:传播行为,read-only:是否只读,是以方法为单位的。这里是将事务管理的通知(这里不用手动书写)织入到业务逻辑形成代理对象。
(6)测试类:
public class Test {
public static void main(String [] args){
ApplicationContext applicationContext=new ClassPathXmlApplicationContext("applicationContext.xml");//创建容器对象
AccountServiceImp accountServiceImp =(AccountServiceImp)applicationContext.getBean("accountserviceimp");
accountServiceImp.transfer(1,2,12d);
}
}
5、Spring事务管理方式:注解
注解:
@Transactional(isolation = Isolation.REPEATABLE_READ,propagation=Propagation.REQUIRED,readOnly = false)
public void transfer(final Integer from, final Integer to,final Double money) {
accountDao.decreaseMoney(from,money);
accountDao.increaseMoney(to,money);
}
配置文件:
<!--指定要读取的配置文件的位置-->
<context:property-placeholder location="classpath:db.properties"></context:property-placeholder>
<!--将连接池放入Spring容器-->
<bean name="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"></property>
<property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
<property name="user" value="${jdbc.user}"></property>
<property name="password" value="${jdbc.password}"></property>
</bean>
<!--Dao-->
<bean name="accountDao" class="pers.zhb.dao.AccountDaoImp">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--Service-->
<bean name="accountserviceimp" class="pers.zhb.service.AccountServiceImp">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!--核心事务管理器,依赖于连接池-->
<bean name="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--事务模板对象-->
<bean name="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager" ref="transactionManager"></property>
</bean>
<!-- 开启使用注解管理aop事务 -->
<tx:annotation-driven/>
<!--java.lang.ClassCastException: com.sun.proxy.$Proxy2 cannot be cast to...异常-->
<aop:aspectj-autoproxy proxy-target-class="true"/>
</beans>