一、spring 事务简介
Spring 本身并不实现事务,Spring事务的本质,还是底层数据库对事务的支持,没有 数据库事务的支持,Spring事务就不会生效。
spring 事务中指的提交和回滚实际上是对数据库的提交和回滚操作。
spring 事务说白了就是简化了操作数据库事务的步骤:
-
获取连接 Connection con = DriverManager.getConnection() -
开启事务con.setAutoCommit(true/false); -
执行CRUD -
提交事务/回滚事务 con.commit() / con.rollback(); -
关闭连接 conn.close();
- 使用spring 为我们提供的事务操作:
Spring事务的本质 其实就是 AOP 和 数据库事务,Spring 将数据库的事务操作提取为 切面,通过AOP的方式 增强 事务方法。 当然具体的操作方式可以分为三种,它们都能实现事务:
- Spring的编程式的事务管理(说白了就是通过存纯java代码实现)
- 使用XML配置声明式事务(通过配置文件spring.xml 来实现)
- 使用注解配置声明式事务(使用注解实现,这种方式最简单,只需要一个注解就能实现,推荐使用)
注意:只讲使用,想知道spring 怎么实现事务的可以观看这篇文章:Spring 事务实现分析
二、几个概念
1、PlatformTransactionManager事务管理器(平台事务管理器)
Spring并不直接管理事务,而是提供了多种事务管理器,它们将事务管理的职责委托给JTA或其他持久化机制所提供的平台相关的事务实现。每个事务管理器都会充当某一特定平台的事务实现的门面,这使得用户在Spring中使用事务时,几乎不用关注实际的事务实现是什么。
Spring提供了许多内置事务管理器实现:
-
DataSourceTransactionManager:位于org.springframework.jdbc.datasource包中,数据源事务管理器,提供对单个javax.sql.DataSource事务管理,用于Spring JDBC抽象框架、iBATIS或MyBatis框架的事务管理; -
JdoTransactionManager:位于org.springframework.orm.jdo包中,提供对单个javax.jdo.PersistenceManagerFactory事务管理,用于集成JDO框架时的事务管理; -
JpaTransactionManager:位于org.springframework.orm.jpa包中,提供对单个javax.persistence.EntityManagerFactory事务支持,用于集成JPA实现框架时的事务管理; -
HibernateTransactionManager:位于org.springframework.orm.hibernate3包中,提供对单个org.hibernate.SessionFactory事务支持,用于集成Hibernate框架时的事务管理;该事务管理器只支持Hibernate3+版本,且Spring3.0+版本只支持Hibernate 3.2+版本; -
JtaTransactionManager:位于org.springframework.transaction.jta包中,提供对分布式事务管理的支持,并将事务管理委托给Java EE应用服务器事务管理器; -
OC4JjtaTransactionManager:位于org.springframework.transaction.jta包中,Spring提供的对OC4J10.1.3+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持; -
WebSphereUowTransactionManager:位于org.springframework.transaction.jta包中,Spring提供的对WebSphere 6.0+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持; -
WebLogicJtaTransactionManager:位于org.springframework.transaction.jta包中,Spring提供的对WebLogic 8.1+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持。
举例:
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
2、TransactionStatus 事务状态
TransactionStatus代表一个事务的具体运行状态。事务管理器通过该接口获取事务的运行期的状态信息。 该接口继承SavepointManager接口。
- SavepointManager接口拥有以下的方法
-
Object createSavepoint():创建一个保存点对象(回滚点)。 -
void rollbackToSavepoint(Object savepoint):回滚到指定的保存点 -
void releaseSavepoint(Object savepoint):释放一个保存点。如果事务提交,所有保存点会自动释放。
- TransactionStatus扩展了SavepointManager并提供了以下的方法
-
boolean hasSavepoint():判断当前的事务是否有保存点。 -
boolean isNewTransaction():判断当前的事务是否是一个新的事务。 -
boolean isCompleted():判断当前事务是否已经结束:已经提交或者回滚 -
boolean isRollbackOnly():判断当前事务是否已经被标识为rollback-only -
void setRollbackOnly():将当前事务设置为rollback-only,通过该标识通知事务管理器只能进行事务回滚。
3、TransactionDefinition 事务属性(重要)
3.1、事务隔离级别(isolation)
- 默认(DEFAULT)
使用底层数据存储的默认隔离级别。 所有其他级别对应于 JDBC 隔离级别
@Transactional(isolation = Isolation.DEFAULT)
<tx:method name="transfer" isolation="DEFAULT" />
- 读未提交(READ_UNCOMMITTED):
指示可能发生脏读、不可重复读和幻读的常量。 此级别允许由一个事务更改的行在提交该行中的任何更改之前被另一个事务读取(“脏读”)。 如果任何更改被回滚,则第二个事务将检索到无效行。读未提交(顾名思义,一个事务读到另一个事务没有提交的数据(脏数据),这个过程交脏读。读未提交不能解决任何事务中出现的问题。)
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
<tx:method name="transfer" isolation="READ_UNCOMMITTED" />
- 读已提交(READ_COMMITTED):
解决脏读; 可能发生不可重复读和幻读。 此级别仅禁止事务读取其中包含未提交更改的行。
@Transactional(isolation = Isolation.READ_COMMITTED)
<tx:method name="transfer" isolation="READ_COMMITTED" />
- 可重复读(REPEATABLE_READ):
表示防止脏读和不可重复读; 可能会发生幻读。 此级别禁止事务读取未提交更改的行,也禁止一个事务读取一行,第二个事务更改该行,第一个事务重新读取该行,第二个获取不同值的情况时间(“不可重复读取”)。 不可重复读一般由修改数据造成。
@Transactional(isolation = Isolation.REPEATABLE_READ)
<tx:method name="transfer" isolation="REPEATABLE_READ" />
- 串行(SERIALIZABLE):
指示防止脏读、不可重复读和幻读的常量。 该级别包括ISOLATION_REPEATABLE_READ的禁止,并进一步禁止这样一种情况:一个事务读取满足WHERE条件的所有行,第二个事务插入满足WHERE条件的行,第一个事务为相同条件重新读取,检索额外的“第二次阅读中的幻影”行。 幻读一般由新增、删除造成。
@Transactional(isolation = Isolation.SERIALIZABLE)
<tx:method name="transfer" isolation="SERIALIZABLE" />
3.2、事务管理器(transactionManager)
事务管理器的概念已经上面已经解释了。事务管理器需要与我们操作数据库的方式一致(事务管理器,顾名思义对我们数据库事务进行管理,也就是事务的提交和回滚,但是不同的数据库操作方式(JDBC抽象框架、MyBatis、Hibernate)对事务的提交、回滚等操作的封装方式不同所有需要指定不同的事务管理器进行管理事务),也需要我们指定和配置的。如下:
<bean id = "datasource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${driverClass}"></property>
<property name="jdbcUrl" value="${jdbcUrl}"></property>
<property name="user" value="${user}"></property>
<property name="password" value="${password}"></property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
<tx:annotation-driven transaction-manager="transactionManager" />
3.3、事务传播类型(propagation)
事务的传播性一般用在事务嵌套的场景,比如一个事务方法里面调用了另外一个事务方法,那么两个方法是各自作为独立的方法提交还是内层的事务合并到外层的事务一起提交,这就是需要事务传播机制的配置来确定怎么样执行。 一般来说都是业务层调用数据层,但是也有特殊情况业务层之间相互调用。当业务层相互调用时,该怎么处理业务层之间的事务关系。
spring 事务帮我们封装好了一些常用的处理方式(事务传播类型),如下:
- PROPAGATION_REQUIRED
Spring默认的传播机制,能满足绝大部分业务需求,如果外层有事务,则当前事务加入到外层事务,一块提交,一块回滚。如果外层没有事务,新建一个事务执行。 - PROPAGATION_REQUES_NEW
该事务传播机制是每次都会新开启一个事务,同时把外层事务挂起,当当前事务执行完毕,恢复上层事务的执行。如果外层没有事务,执行当前新开启的事务即可 - PROPAGATION_SUPPORT
如果外层有事务,则加入外层事务,如果外层没有事务,则直接使用非事务方式执行。完全依赖外层的事务 PROPAGATION_NOT_SUPPORT 该传播机制不支持事务,如果外层存在事务则挂起,执行完当前代码,则恢复外层事务,无论是否异常都不会回滚当前的代码 - PROPAGATION_NEVER
该传播机制不支持外层事务,即如果外层有事务就抛出异常 - PROPAGATION_MANDATORY
与NEVER相反,如果外层没有事务,则抛出异常 - PROPAGATION_NESTED
该传播机制的特点是可以保存状态保存点,当前事务回滚到某一个点,从而避免所有的嵌套事务都回滚,即各自回滚各自的,如果子事务没有把异常吃掉,基本还是会引起全部回滚的。
举例:
@Transactional(propagation = Propagation.REQUIRED)
<tx:method name="*" propagation="REQUIRED" />
3.4、超时时间(timeout)
这个很好理解,也就是调用数据层时,等待返回结果的时间。超时就抛出异常,结束调用。 默认为:使用底层事务系统的默认超时,如果不支持超时,则不使用。 单位为:秒 举例:
@Transactional(timeout = -1)
<!--为全部方法配置属性-->
<tx:method name="*" timeout="-1" propagation="REQUIRED" />
3.5、只读(readOnly)
如果事务实际上是只读的,则可以将其设置为true布尔标志,从而允许在运行时进行相应的优化。 默认为false 。 这只是对实际事务子系统的提示; 它不一定会导致写访问尝试失败。 无法解释只读提示的事务管理器在请求只读事务时不会抛出异常,而是默默地忽略该提示。
@Transactional(readOnly = true)
3.6、筛选出现异常哪些后导致事务回滚( rollbackFor、noRollbackFor)
- rollbackFor
定义零 (0) 个或多个异常classes ,它们必须是Throwable子类,指示哪些异常类型必须导致事务回滚。 默认情况下,事务将在RuntimeException和Error上回滚,但不会在已检查的异常(业务异常)上回滚。 - rollbackForClassName
定义零 (0) 个或多个异常名称(对于必须是Throwable子类的异常),指示哪些异常类型必须导致事务回滚。 这可以是完全限定类名的子字符串,目前不支持通配符。 例如, "ServletException"的值将匹配javax.servlet.ServletException及其子类。 注意:仔细考虑模式的具体程度以及是否包含包信息(这不是强制性的)。 例如, “Exception"几乎可以匹配任何内容,并且可能会隐藏其他规则。 如果"Exception"旨在为所有已检查的异常定义规则,则"java.lang.Exception"将是正确的。 对于更不寻常的Exception名称,例如"BaseBusinessException” ,则无需使用 FQN - noRollbackFor
定义零 (0) 个或多个异常Classes ,它们必须是Throwable子类,指示哪些异常类型不得导致事务回滚。 - noRollbackForClassName
定义零 (0) 个或多个异常名称(对于必须是Throwable子类的异常),指示哪些异常类型不得导致事务回滚。
@Transactional(rollbackFor = Exception.class)
@Transactional(rollbackForClassName= Exception.class)
@Transactional(noRollbackFor= IOException.class)
4、其他
4.1、回滚规则
在默认设置下,事务只在出现运行时异常(runtime exception)时回滚,而在出现受检查异常(checked exception)时不回滚(这一行为和EJB中的回滚行为是一致的)。 不过,可以声明在出现特定受检查异常时像运行时异常一样回滚。同样,也可以声明一个事务在出现特定的异常时不回滚,即使特定的异常是运行时异常。
三、使用
导入pom依赖(注意版本,版本不一致会报错)
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.4</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<dependency>
<groupId>com.mchange</groupId>
<artifactId>c3p0</artifactId>
<version>0.9.5.2</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.32</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13.1</version>
</dependency>
</dependencies>
1、编程式(了解即可)
数据层
接口
package com.lihua.spring事务_编程式01.mapper;
public interface BankMapper {
public void addMoney(String name,Double money);
public void reduceMoney(String name,Double money);
}
实现类
package com.lihua.spring事务_编程式01.mapper.impl;
import com.lihua.spring事务_编程式01.mapper.BankMapper;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
public class BankMapperImpl extends JdbcDaoSupport implements BankMapper {
public void addMoney(String name, Double money) {
String sql="update bank_card set money= money+ ? where username=?";
this.getJdbcTemplate().update(sql, money,name);
}
public void reduceMoney(String name, Double money) {
String sql="update bank_card set money= money- ? where username=?";
this.getJdbcTemplate().update(sql, money,name);
}
}
业务层(核心代码)
接口
package com.lihua.spring事务_编程式01.service;
public interface BankService {
void transfer(String payee,String payer,Double money);
}
实现类
package com.lihua.spring事务_编程式01.service.impl;
import com.lihua.spring事务_编程式01.mapper.BankMapper;
import com.lihua.spring事务_编程式01.service.BankService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;
import javax.annotation.Resource;
@Service("bankService")
public class BankServiceImpl implements BankService {
@Resource(name = "template")
private TransactionTemplate template;
@Autowired
private BankMapper bankMapper;
public void transfer(final String payee, final String payer, final Double money) {
template.execute(new TransactionCallbackWithoutResult() {
@Override
protected void doInTransactionWithoutResult(TransactionStatus status) {
bankMapper.addMoney(payee,money);
bankMapper.reduceMoney(payer,money);
}
});
}
}
spring.xml配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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.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
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<context:component-scan base-package="com.lihua.spring事务_编程式01"/>
<context:annotation-config></context:annotation-config>
<context:property-placeholder location="classpath:db.properties"/>
<bean id = "datasource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${driverClass}"></property>
<property name="jdbcUrl" value="${jdbcUrl}"></property>
<property name="user" value="${user}"></property>
<property name="password" value="${password}"></property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
<bean id="template" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager" ref="transactionManager"></property>
</bean>
<bean id="bankDao" class="com.lihua.spring事务_编程式01.mapper.impl.BankMapperImpl">
<property name="dataSource" ref="datasource"></property>
</bean>
</beans>
测试
package com.lihua.spring事务_编程式01.test;
import com.lihua.spring事务_编程式01.service.BankService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
public static void main(String[] args) {
ApplicationContext ac=new ClassPathXmlApplicationContext("spring-transaction1.xml");
BankService service=(BankService) ac.getBean("bankService");
service.transfer("李华", "小明", 200.0);
}
}
2、声明式
数据层
接口
package com.lihua.spring事务_声明式02.mapper;
public interface BankMapper {
public void addMoney(String name, Double money);
public void reduceMoney(String name, Double money);
}
实现类
package com.lihua.spring事务_声明式02.mapper.impl;
import com.lihua.spring事务_声明式02.mapper.BankMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;
@Repository("bankMapper")
public class BankMapperImpl implements BankMapper {
@Autowired
private JdbcTemplate jdbcTemplate;
public void addMoney(String name, Double money) {
String sql="update bank_card set money= money+ ? where username=?";
jdbcTemplate.update(sql, money,name);
}
public void reduceMoney(String name, Double money) {
String sql="update bank_card set money= money- ? where username=?";
jdbcTemplate.update(sql, money,name);
}
}
业务层
接口
package com.lihua.spring事务_声明式02.service;
public interface BankService {
void transfer(String payee, String payer, Double money);
}
实现类
package com.lihua.spring事务_声明式02.service.impl;
import com.lihua.spring事务_声明式02.mapper.BankMapper;
import com.lihua.spring事务_声明式02.service.BankService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Service;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;
import javax.annotation.Resource;
@Service("bankService")
public class BankServiceImpl implements BankService {
@Autowired
@Qualifier("bankMapper")
private BankMapper bankMapper;
public void transfer(final String payee, final String payer, final Double money) {
bankMapper.addMoney(payee,money);
int i =1/0;
bankMapper.reduceMoney(payer,money);
}
public String aa() {
return "123";
}
}
spring.xml配置(核心代码)
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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.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
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<context:component-scan base-package="com.lihua.spring事务_声明式02"/>
<context:annotation-config></context:annotation-config>
<context:property-placeholder location="classpath:db.properties"/>
<bean id = "datasource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${driverClass}"></property>
<property name="jdbcUrl" value="${jdbcUrl}"></property>
<property name="user" value="${user}"></property>
<property name="password" value="${password}"></property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="datasource"></property>
</bean>
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="transfer" isolation="DEFAULT" read-only="false" />
<tx:method name="aa" isolation="READ_COMMITTED" read-only="false" />
<tx:method name="*" timeout="-1" propagation="REQUIRED" />
</tx:attributes>
</tx:advice>
<aop:config>
<aop:advisor advice-ref="txAdvice" pointcut="execution(* com.lihua.spring事务_声明式02.service.impl.BankServiceImpl.*(..))" />
</aop:config>
</beans>
测试
package com.lihua.spring事务_声明式02.test;
import com.lihua.spring事务_声明式02.service.BankService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
public static void main(String[] args) {
ApplicationContext ac=new ClassPathXmlApplicationContext("spring-transaction2.xml");
BankService proxyBankService=(BankService) ac.getBean("bankService");
proxyBankService.transfer("李华", "小明", 200.0);
System.out.println("转账成功");
}
}
3、注解式
数据层(不变)
接口
package com.lihua.spring事务_注解式03.mapper;
public interface BankMapper {
public void addMoney(String name, Double money);
public void reduceMoney(String name, Double money);
}
实现类
package com.lihua.spring事务_注解式03.mapper.impl;
import com.lihua.spring事务_注解式03.mapper.BankMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;
@Repository("bankMapper")
public class BankMapperImpl implements BankMapper {
@Autowired
@Qualifier("jdbcTemplate")
private JdbcTemplate jdbcTemplate;
public void addMoney(String name, Double money) {
String sql="update bank_card set money= money+ ? where username=?";
jdbcTemplate.update(sql, money,name);
}
public void reduceMoney(String name, Double money) {
String sql="update bank_card set money= money- ? where username=?";
jdbcTemplate.update(sql, money,name);
}
}
业务层(核心代码)
接口
package com.lihua.spring事务_注解式03.service;
public interface BankService {
void transfer(String payee, String payer, Double money);
}
实现类
package com.lihua.spring事务_注解式03.service.impl;
import com.lihua.spring事务_注解式03.mapper.BankMapper;
import com.lihua.spring事务_注解式03.service.BankService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Service("bankService1")
public class BankServiceImpl implements BankService {
@Autowired
@Qualifier("bankMapper")
private BankMapper bankMapper;
@Transactional(
isolation = Isolation.DEFAULT,
timeout = 3,
readOnly = false,
rollbackFor = Exception.class
)
public void transfer(final String payee, final String payer, final Double money) {
bankMapper.addMoney(payee,money);
int i =1/0;
bankMapper.reduceMoney(payer,money);
}
}
spring.xml配置(更改配置)
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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.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
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<context:component-scan base-package="com.lihua.spring事务_注解式03"/>
<context:annotation-config></context:annotation-config>
<context:property-placeholder location="classpath:db.properties"/>
<bean id = "datasource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${driverClass}"></property>
<property name="jdbcUrl" value="${jdbcUrl}"></property>
<property name="user" value="${user}"></property>
<property name="password" value="${password}"></property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"></property>
</bean>
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="datasource"></property>
</bean>
<tx:annotation-driven transaction-manager="transactionManager" />
</beans>
测试
package com.lihua.spring事务_注解式03.test;
import com.lihua.spring事务_注解式03.service.BankService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
public static void main(String[] args) {
ApplicationContext ac=new ClassPathXmlApplicationContext("spring-transaction3.xml");
BankService bankService = (BankService)ac.getBean("bankService1");
bankService.transfer("李华", "小明", 200.0);
System.out.println("成功");
}
}
4、ssm框架使用事务
使用与上面的注解式差不多(一样),
4.1、导入pom依赖(注意版本问题)
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13.1</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.5</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.47</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.2.5</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.8</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.3.8</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.3.4</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.3.4</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.3.8</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.6</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.3.3</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.2</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.3.8</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.51</version>
</dependency>
</dependencies>
4.2、spring配置
在整合ssm spring配置的基础上加上下面的配置即可
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager" />
4.3、通过注解使用事务
package com.lihua.ssm.service.impl;
import com.lihua.ssm.mapper.TransferMapper;
import com.lihua.ssm.pojo.BankCard;
import com.lihua.ssm.service.TransferService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
@Service
public class TransferServiceImpl implements TransferService {
@Autowired
private TransferMapper transferMapper;
@Transactional(isolation = Isolation.READ_COMMITTED,rollbackFor =Exception.class )
public boolean transfer(String payee, String payer, Double money) {
transferMapper.addMoney(payee,money);
int i =1/0;
transferMapper.reduceMoney(payer,money);
System.out.println("转账成功");
return true;
}
public List<BankCard> queryBankCard() {
return transferMapper.queryBankCard();
}
}
5、spring boot使用事务
5.1、导入pom依赖
在Spring Boot中,当我们使用了spring-boot-starter-jdbc或spring-boot-starter-data-jpa依赖的时候,框 架会自动默认分别注入DataSourceTransactionManager或JpaTransactionManager。所以我们不需要任何额外 配置就可以用@Transactional注解进行事务的使用。
5.2、使用
spring Boot 使用事务非常简单,首先使用注解 @EnableTransactionManagement 开启事务支持后,然后在访问数据库的Service方法上添加注解 @Transactional 便可。
主启动类
package com.lihua.newideas;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity;
import org.springframework.transaction.annotation.EnableTransactionManagement;
@SpringBootApplication
@EnableTransactionManagement(proxyTargetClass = true)
@EnableGlobalMethodSecurity(securedEnabled = true , prePostEnabled = true)
public class NewIdeasApplication {
public static void main(String[] args) {
SpringApplication.run(NewIdeasApplication.class, args);
}
}
使用注解开始事务
@Transactional
@Override
public String updateUserByUserName(String passWord, String userName, String code) {
MyUser myuser = queryUserByName(userName);
if(myuser==null){
return "没有这个用户";
}else {
if(myuser.getCode().equals(code)){
String encode = passwordEncoder.encode(passWord);
myUserMapper.updateUserByUserName(encode,userName,code);
int i =1/0;
return "修改成功";
}else {
return "密保错误";
}
}
}
5.3、手动提交回滚事务
spring boot事务回滚(出现运行时异常)与提交(没有异常)都是自动完成的我们也可以手动操作。 具体可以参考: SpringBoot声明式事务的简单运用
|