Spring中事务管理的四种方法(银行转账为例)

网友投稿 335 2023-02-04


Spring中事务管理的四种方法(银行转账为例)

前言

本文配套示例代码下载地址(完整可运行,含sql文件,下载后请修改数据库配置):点击这里下载

一、事务的作用

将若干的数据库操作作为一个整体控制,一起成功或一起失败。

原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。

一致性:指事务前后数据的完整性必须保持一致。

隔离性:指多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离。

持久性:指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,即时数据库发生故障也不应该对其有任何影响。

二、Spring事务管理高层抽象主要包括3个接口

--Platform TransactionManager 事务管理器(提交、回滚事务)

Spring为不同的持久化框架提供了不同的Platform TransactionManager接口实现。如:

使用Spring JDBC或iBatis进行持久化数据时使用DataSourceTransactionManager

使用Hibernate3.0版本进行持久化数据时使用HibernateTransactionManager

--TransactionDefinition 事务定义信息(隔离、传播、超时、只读)

脏读:一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。

不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同。

幻读:一个事务读取了几行记录后,另一个事务插入一些记录,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。

事务隔离级别:(五种)

DEFAULT--使用后端数据库默认的隔离级别(Spring中的选择项)

    READ_UNCOMMITED--允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读

    READ_COMMITTED--允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生

    REPEATABLE_READ--对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生

    SERIALIZABLE--完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的

其中,mysql默认采用REPEATABLE_READ隔离级别;Oracle默认采用READ_COMMITTED隔离级别

       事务传播行为:(七种)

REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。

    SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。

    MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。

    REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。

    NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。

    NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。

    NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与REQUIRED类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对DataSourceTransactionManager有效

--TransactionStatus 事务具体运行状态

三、Spring提供了以下方法控制事务

a.编程式事务管理(基于java编程控制,很少使用)--见demo1包

利用TransactionTemplate将多个DAO操作封装起来

*b.声明式事务管理(基于Spring的AOP配置控制)

-基于TransactionProxyFactoryBean的方式.(很少使用)--见demo2包

需要为每个进行事务管理的类,配置一个TransactionProxyFactoryBean进行增强.

-基于XML配置(经常使用)--见demo3包

一旦配置好之后,类上不需要添加任何东西。

如果Action作为目标对象切入事务,需要在元素里添加proxy-target-class="true"属性。原因是通知Spring框架采用CGLIB技术生成具有事务管理功能的Action类。

-基于注解(配置简单,经常使用)--见demo4包

在applicationContext.xml中开启事务注解配置。(applicationContext.xml中只需定义Bean并追加以下元素)

在目标组件类中使用@Transactional,该标记可定义在类前或方法前。

四、示例(银行转账)

--编程式

/**

* @Description:转账案例的DAO层接口

*

*/

public interface AccountDao {

/**

* @param out

* :转出账号

* @param money

* :转账金额

*/

public void outMoney(String out, Double money);

/**

*

* @param in

* :转入账号

* @param money

* :转账金额

*/

public void inMoney(String in, Double money);

}

/**

* @Description:转账案例的DAO层实现类

*/

public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {

/**

* @param out

* :转出账号

* @param money

* :转账金额

*/

@Override

public void outMoney(String out, Double money) {

String sql = "update account set money = money-? where name = ?";

this.getJdbcTemplate().update(sql, money, out);

}

/**

* @param in

* :转入账号

* @param money

* :转账金额

*/

@Override

public void inMoney(String in, Double money) {

String sql = "update account set money = money+? where name = ?";

this.getJdbcTemplate().update(sql, money, in);

}

}

/**

* @Description:转账案例的业务接口

*

*/

public interface AccountService {

/**

* @param out :转出账号

* @param in :转入账号

* @param money :转账金额

*/

public void transfer(String out,String in,Double money);

}

/**

* @Description:转账案例的业务层实现类

*/

public class AccountServiceImpl implements AccountService {

// 注入转账的DAO

private AccountDao accountDao;

// 注入事务管理的模板

private TransactionTemplate transactionTemplate;

/**

* @param out

* :转出账号

* @param in

* :转入账号

* @param money

* :转账金额

*/

@Override

public void transfer(final String out, final String in, final Double money) {

// 未经事务控制的业务处理操作,如果过程中出异常,则导致前面的操作能完成,后面的不能,即转账成功但未收到转账款

// accountDao.outMoney(out, money);

// int i = 1/0;

// accountDao.inMoney(in, money);

transactionTemplate.execute(new TransactionCallbackWithoutResult() {

@Override

protected void doInTransactionWithoutResult(

TransactionStatus transactionStatus) {

accountDao.outMoney(out, money);

// int i = 1 / 0;//事务控制,即出现异常,该段内代码都执行失效

accountDao.inMoney(in, money);

}

});

}

public void setAccountDao(AccountDao accountDao) {

this.accountDao = accountDao;

}

public void setTransactionTemplate(TransactionTemplate transactionTemplate) {

this.transactionTemplate = transactionTemplate;

}

}

applicationContext1.xml

测试:

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration("classpath:applicationContext1.xml")

public class TransactionTest {

@Resource(name = "accountService")

private AccountService accountService;

@Test

public void demo1() {

accountService.transfer("aaa", "bbb", 200d);

}

}

--基于TransactionProxyFactoryBean的方式

public class AccountServiceImpl implements AccountService {

// 注入转账的DAO

private AccountDao accountDao;

/**

* @param out

* :转出账号

* @param in

* :转入账号

* @param money

* :转账金额

*/

@Override

public void transfer(String out, String in, Double money) {

accountDao.outMoney(out, money);

// int i = 1/0;

accountDao.inMoney(in, money);

}

public void setAccountDao(AccountDao accountDao) {

this.accountDao = accountDao;

}

}

applicationContext2.xml

PROPAGATION_REQUIRED

测试:

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration("classpath:applicationContext2.xml")

public class TransactionTest {

/**

* 一定要注入代理类:因为代理类进行增强的操作

*/

// @Resource(name="accountService")

@Resource(name = "accountServiceProxy")

private AccountService accountService;

@Test

public void demo1() {

accountService.transfer("aaa", "bbb", 200d);

}

}

--基于XML配置

public class AccountServiceImpl implements AccountService {

// 注入转账的DAO

private AccountDao accountDao;

/**

* @param out

* :转出账号

* @param in

* :转入账号

* @param money

* :转账金额

*/

@Override

public void transfer(String out, String in, Double money) {

accountDao.outMoney(out, money);

// int i = 1/0;

accountDao.inMoney(in, money);

}

public void setAccountDao(AccountDao accountDao) {

this.accountDao = accountDao;

}

}

applicationContext3.xml

测试:

/**

* @Description:Spring的声明式事务管理的方式二:基于AspectJ的XML方式的配置

*/

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration("classpath:applicationContext3.xml")

public class TransactionTest {

/**

* 一定要注入代理类:因为代理类进行增强的操作

*/

@Resource(name = "accountService")

private AccountService accountService;

@Test

public void demo1() {

accountService.transfer("aaa", "bbb", 200d);

}

}

--基于注解

/**

* @Transactional中的的属性 propagation :事务的传播行为 isolation :事务的隔离级别 readOnly :只读

* rollbackFor :发生哪些异常回滚 noRollbackFor :发生哪些异常不回滚

* rollbackForClassName 根据异常类名回滚

*/

@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)

public class AccountServiceImpl implements AccountService {

// 注入转账的DAO

private AccountDao accountDao;

/**

* @param out

* :转出账号

* @param in

* :转入账号

* @param money

* :转账金额

*/

@Override

public void transfer(String out, String in, Double money) {

accountDao.outMoney(out, money);

// int i = 1/0;

accountDao.inMoney(in, money);

}

public void setAccountDao(AccountDao accountDao) {

this.accountDao = accountDao;

}

}

applicationContext4.xml

测试:

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration("classpath:applicationContext4.xml")

public class TransactionTest {

/**

* 一定要注入代理类:因为代理类进行增强的操作

*/

@Resource(name = "accountService")

private AccountService accountService;

@Test

public void demo1() {

accountService.transfer("aaa", "bbb", 200d);

}

}

具体代码和数据库文件参考项目完整代码:

http://xiazai.jb51.net/201805/yuanma/Spring-transaction_jb51.rar

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:跨局域网连接共享文件夹(跨局域网连接共享文件夹怎么弄)
下一篇:java实现文件编码转换的方法
相关文章

 发表评论

暂时没有评论,来抢沙发吧~