首先,什么是Spring的事务管理?
在实际开发中,操作数据库时都会涉及到事务管理问题,为此Spring提供了专门用于事务处理的API。Spring的事务管理简化了传统的事务管理流程,并且在一定程度上减少了开发者的工作量。
在Spring的所有JAR包中,包含一个名为spring-tx-4.3.6.RELEASE的JAR包,该包就是Spring提供的用于事务管理的依赖包。在该JAR包的org.springframework.transaction包中,有3个接口文件PlatformTransactionManager、TransactionDefinition和TransactionStatus.
然后,如何实现Spring的声明式事务管理?
Spring的声明式事务管理可以通过两种方式来实现,一种是基于XML的方式,另一种是基于Annotation的方式。接下来的两个小节中,将对这两种声明式事务管理方式进行详细讲解。
1.XML来实现,例子如下:
(1)创建Account类,添加id,username,balance成员变量和各自的get,set方法.
(2)声明AccountDao接口和AccountDaoImpl实现类
public class AccountDaoImpl implements AccountDao {
private JdbcTemplate jdbcTemplate;
public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
this.jdbcTemplate = jdbcTemplate;
}
@Override
public void transfer(String outUser, String inUser, Double money) {
this.jdbcTemplate.update("update account set balance = balance + ? " +
"where username = ?",money,inUser);
int i = 1 / 0;
this.jdbcTemplate.update("update account set balance = balance - ? " +
"where username = ?",money,outUser);
}
}
(3)xml配置:
<bean id="dataSource"
class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/dangdang"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</bean>
<!-- JDBC模版-->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"/>
</bean>
<bean id="accountDao" class="com.itheima.jdbc.AccountDaoImpl">
<property name="jdbcTemplate" ref="jdbcTemplate"/>
</bean>
<bean id="account" class="com.itheima.jdbc.Account">
</bean>
(4)测试类:
public class TransactionTest {
public static void main(String[] args) {
ApplicationContext application =
new ClassPathXmlApplicationContext("applicationContext.xml");
AccountDao accountDao = (AccountDao) application.getBean("accountDao");
accountDao.transfer("Jack", "Rose", 100.0);
System.out.println("转账成功!");
}
}
运行结果如下:
由于AccountDaoImpl的int i = 1/0;程序报错没什么问题,但是,你到数据库里会发现Rose的钱多了,但是Jack的钱并没有减少!
(原余额各有1000)
为什么?因为转钱的操作分为让转出者减少余额和让接受者增加余额,上面程序中增加余额的操作刚执行完,程序就报错,还没来得及执行减钱的指令,程序就GG了,这样子显然是不科学的.(银行得赔死),接下来我们对程序加以改进.
1>xml中添加以下代码
<!-- 事务管理器,依赖于数据源 -->
<bean id="transactionManager" class=
"org.springframework.jdbc.datasource.DataSourceTransactionManager" >
<property name="dataSource" ref="dataSource"/>
</bean>
<!--写通知:对事务进行增强,需要编写对切入点和具体执行事务细节-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" propagation="REQUIRED"
isolation="DEFAULT" read-only="false"/>
</tx:attributes>
</tx:advice>
<!--编写aop,让spring自动对目标生成代理,需要使用AspectJ的表达式-->
<aop:config>
<!-- 切入点 -->
<aop:pointcut id="txPointCut" expression="execution(* com.itheima.jdbc..*.*(..))"/>
<!-- 切面:将切入点与通知整合-->
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointCut"/>
</aop:config>
此时你会发现运行测试类还是会报错,但是,余额总数是合理的.这说明事务管理已经生效.
2>我们还可以通过注解的方式,来实现事务管理.
将上述的xml中添加的代码,改为
<!-- 事务管理器,依赖于数据源 -->
<bean id="transactionManager" class=
"org.springframework.jdbc.datasource.DataSourceTransactionManager" >
<property name="dataSource" ref="dataSource"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>
在AccountDaoImpl中transfer上面添加注解
@Transactional(propagation = Propagation.REQUIRED,
isolation = Isolation.DEFAULT,readOnly = false)
@Override
public void transfer(String outUser, String inUser, Double money) {
此时你会发现事务管理也生效了,和xml方式的配置文件相比,后者通过事务管理器驱动,替换了编写通知和编写aop,大大的减少了代码量,需要注意的是,如果案例中使用了注解式开发,则需要在配置文件中开启注解处理器,指定扫描哪些包下的注解.这里没有开启注解处理器是因为在配置文件中已经配置了AccountDaoImpl类的Bean,而@Transactional直接就配置在该Bean类中,左移可以直接生效.