作者简介 :☕️大家好,我是Aomsir,一个爱折腾的开发者!
个人主页 :Aomsir_Spring5应用专栏,Netty应用专栏,RPC应用专栏
当前专栏 :Spring5应用专栏_Aomsir的博客
参考文献
前言
上一篇文章的尾声,我们简要触及了**"事务"**这一重要概念。事务在数据库操作中扮演着至关重要的角色,确保数据的完整性与一致性。今天,我将带领大家进一步深入,系统性地探索Spring是如何进行事务管理和控制的。
事务管理在Spring框架中占据着核心的位置,它为我们提供了强大而灵活的事务控制机制,无论是声明式还是编程式。为了让大家能够从中受益,我们会从事务的基本概念开始,然后深入到Spring的事务管理策略,以及如何在实际应用中恰当地使用这些策略。
希望通过今天的深入学习,大家能够更为熟练地在Spring中操作事务,确保业务逻辑的健壮性和数据的安全性
事务处理
什么是事物?
事务是数据库管理系统中为了确保数据完整性与一致性所提供的一种关键机制。它确保了由一个或多个SQL语句组成的操作作为一个整体被执行,即要么全部成功执行,要么全部不执行。这种全体性的执行特性,保障了即使在面对系统故障或其他意外情况时,数据仍然保持一致性。
事务的四大特性,通常被称为ACID属性,具体包括:
- 原子性 (Atomicity):原子性确保事务作为一个单一的"单位"进行操作,意味着如果事务中的某些操作失败,整个事务都将回滚,就好像从未被执行过一样。
- 隔离性 (Isolation):多个事务并发执行时,隔离性确保每个事务在一个单独的"隔离"环境中运行,使得事务之间不会相互影响。
- 一致性 (Consistency):一致性确保每个事务执行完后,数据库的状态从一个一致的状态转变为另一个一致的状态。
- 持久性 (Durability):持久性保证一旦事务被提交,对数据库的更改就是永久性的,即使系统发生故障。
了解这四大特性对于理解事务的工作原理和如何在实际应用中使用事务至关重要。在使用Spring进行数据库操作时,Spring提供了强大的事务管理功能,允许开发者方便地控制和管理事务,从而确保数据操作的完整性和一致性
如何控制事务?
事务管理在数据库操作中是至关重要的,不同的框架或库为我们提供了不同的事务控制手段,但本质上,这些操作都是基于数据库连接对象Connection的。
- JDBC :在传统的JDBC编程中,事务的控制是直接通过Connection对象来实现的。
Connection.setAutoCommit(false)
可以关闭自动提交,这样可以确保在明确调用commit()
方法之前,所做的更改不会被提交。如果在事务中遇到错误,可以通过rollback()
方法来撤回更改。这种手动控制事务的方式给予开发者很大的灵活性,但同时也带来了更多的责任,因为开发者需要确保在正确的时机提交或回滚事务。 - MyBatis :与JDBC不同,MyBatis为开发者提供了一个更高级的抽象,即SqlSession 。尽管SqlSession为我们提供了诸如
commit()
和rollback()
等方法,但实际上,这些方法的工作仍然是基于底层的Connection对象来完成的。也就是说,SqlSession只是为Connection提供了一个简化的接口,将更复杂的操作隐藏起来,为开发者提供了一个更简洁、更直观的API。
这两种方式各有优缺点。直接使用Connection对象进行事务管理为开发者提供了最大的灵活性,但也需要开发者对事务有深入的理解,确保事务在正确的时机被提交或回滚。而通过SqlSession进行事务管理,虽然部分隐藏了底层细节,但为开发者提供了一个更为简单和直观的API,有助于简化代码并减少出错的机会
Spring控制事务开发
思路分析
在Spring框架中,事务管理是通过AOP技术实现的,它提供了一种高效、灵活的方法来管理事务。事务管理在Spring中的实现与我们之前在AOP中讨论的基本概念紧密相连:原始对象、额外功能、切入点和组装切面。
- 原始对象:在事务管理的上下文中,原始对象通常指的是业务逻辑类,如Service对象。这些对象中包含核心业务逻辑、计算和对DAO的调用。
- 额外功能 :这是真正实现事务管理的部分。当我们调用一个业务方法时,Spring会自动开启一个新的事务。在业务逻辑执行完成之后,Spring会决定提交事务或在出现异常时回滚事务。为了完成这个任务,Spring提供了
DataSourceTransactionManager
类。这个类为我们处理了大部分与事务管理相关的复杂性,如事务的开启、提交和回滚等。 - 切入点 :在Spring中,我们可以通过在方法或类上添加
@Transactional
注解来标识一个切入点。这告诉Spring哪些方法需要事务管理。当标注了此注解的方法被调用时,Spring会自动为其提供事务管理服务。 - 组装切面 :在Spring的配置中,我们使用
<tx:annotation-driven>
标签来启用基于注解的事务管理。此标签告诉Spring去查找带有@Transactional注解的方法,并为其提供事务管理服务。为了使这一切正常工作,我们还需要提供一个DataSourceTransactionManager的引用,它会负责实际的事务管理工作
编码分析
步骤如下
- 引入依赖
- 开发原始业务Service类,添加注解
- 配置文件配置
xml
<!--Spring事务控制所需依赖-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.1.14.RELEASE</version>
</dependency>
java
@Transactional
public class UserServiceImpl implements UserService {
private UserDAO userDAO;
public UserDAO getUserDAO() {
return userDAO;
}
public void setUserDAO(UserDAO userDAO) {
this.userDAO = userDAO;
}
@Override
public void register(User user) {
userDAO.save(user);
throw new RuntimeException("测试异常");
}
}
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: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/tx https://www.springframework.org/schema/tx/spring-tx.xsd">
<!--此处省略内容为上一节Spring整合的内容-->
<!--原始对象-->
<bean id="userService" class="com.aomsir.basic.mybatis.service.impl.UserServiceImpl">
<property name="userDAO" ref="userDAO" />
</bean>
<!--额外功能-->
<bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!--组装切面-->
<tx:annotation-driven transaction-manager="dataSourceTransactionManager" />
</beans>
java
public class TestSpringTx {
@Test
public void test1() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext3.xml");
UserService userService = (UserService) ctx.getBean("userService");
User user = new User();
user.setName("Aomsir");
user.setPassword("123456");
userService.register(user);
}
}
细节分析
确实,tx:annotation-driven
标签的proxy-target-class属性
为我们提供了一个选择代理实现的机会,进一步印证了Spring事务管理的AOP基础。
- JDK动态代理:当proxy-target-class属性设置为false(或未明确设置,因为默认为false)时,Spring使用JDK的动态代理来创建代理对象。JDK动态代理的主要要求是目标类必须实现一个或多个接口,因为代理类也会实现这些接口。因此,只有实现了接口的业务bean才能使用JDK动态代理。
- Cglib动态代理:当proxy-target-class属性设置为true时,Spring使用Cglib库创建代理对象。与JDK动态代理不同,Cglib可以为没有实现任何接口的类创建代理,因为它通过继承目标类的方式创建子类作为代理。这使得Cglib对那些未实现接口的业务bean更加有用。
选择哪种代理方式取决于具体的应用场景和需求。如果业务对象实现了接口,JDK动态代理可能是首选,因为它通常有更好的性能。但是,如果业务对象没有实现任何接口或者有某些原因需要使用Cglib,那么就可以选择Cglib动态代理。 总之,事实上Spring通过proxy-target-class属性提供了对事务管理代理方式的控制,进一步证明了其事务管理是基于AOP的实现,无论是通过JDK动态代理还是Cglib动态代理,都是遵循AOP的核心原则和概念
xml
<!--组装切面-->
<tx:annotation-driven transaction-manager="dataSourceTransactionManager" proxy-target-class="true"/>
总结
在这篇《Spring5应用之事务处理》中,我们详细探讨了Spring中事务的关键概念和实现机制。首先,我们深入理解了事务的核心特性AICD,并通过对比JDBC和MyBatis在事务处理上的实现,进一步明确了事务控制的细节和特点。重要的是,我们发现Spring通过AOP技术巧妙地进行事务管理,从而大大简化了开发者的工作。
文章强调了原始对象如何与额外功能相结合,以及如何通过特定的注解和配置实现事务控制。特别是通过深入研究tx:annotation-driven标签,我们进一步了解了Spring在事务处理背后的AOP实现机制,如何选择不同的代理方法来达到我们的需求。
总而言之,本文为我们提供了一个全面、系统的视角,看待Spring在事务处理上的优越性和实现方法,帮助我们更加自信地在实际开发中应用Spring进行事务管理