作者简介:☕️大家好,我是Aomsir,一个爱折腾的开发者!
个人主页:Aomsir_Spring5应用专栏,Netty应用专栏,RPC应用专栏-CSDN博客
当前专栏:Spring5应用专栏_Aomsir的博客-CSDN博客
- 孙哥suns说Spring5~学不会Spring? 因为你没找对人~孙帅
- Spring官方文档
上一篇文章的尾声,我们简要触及了“事务”这一重要概念。事务在数据库操作中扮演着至关重要的角色,确保数据的完整性与一致性。今天,我将带领大家进一步深入,系统性地探索Spring是如何进行事务管理和控制的。
事务管理在Spring框架中占据着核心的位置,它为我们提供了强大而灵活的事务控制机制,无论是声明式还是编程式。为了让大家能够从中受益,我们会从事务的基本概念开始,然后深入到Spring的事务管理策略,以及如何在实际应用中恰当地使用这些策略。
希望通过今天的深入学习,大家能够更为熟练地在Spring中操作事务,确保业务逻辑的健壮性和数据的安全性
事务是数据库管理系统中为了确保数据完整性与一致性所提供的一种关键机制。它确保了由一个或多个SQL语句组成的操作作为一个整体被执行,即要么全部成功执行,要么全部不执行。这种全体性的执行特性,保障了即使在面对系统故障或其他意外情况时,数据仍然保持一致性。
事务的四大特性,通常被称为ACID属性,具体包括:
了解这四大特性对于理解事务的工作原理和如何在实际应用中使用事务至关重要。在使用Spring进行数据库操作时,Spring提供了强大的事务管理功能,允许开发者方便地控制和管理事务,从而确保数据操作的完整性和一致性
事务管理在数据库操作中是至关重要的,不同的框架或库为我们提供了不同的事务控制手段,但本质上,这些操作都是基于数据库连接对象Connection的。
Connection.setAutoCommit(false)
可以关闭自动提交,这样可以确保在明确调用commit()
方法之前,所做的更改不会被提交。如果在事务中遇到错误,可以通过rollback()
方法来撤回更改。这种手动控制事务的方式给予开发者很大的灵活性,但同时也带来了更多的责任,因为开发者需要确保在正确的时机提交或回滚事务。commit()
和rollback()
等方法,但实际上,这些方法的工作仍然是基于底层的Connection对象来完成的。也就是说,SqlSession只是为Connection提供了一个简化的接口,将更复杂的操作隐藏起来,为开发者提供了一个更简洁、更直观的API。这两种方式各有优缺点。直接使用Connection对象进行事务管理为开发者提供了最大的灵活性,但也需要开发者对事务有深入的理解,确保事务在正确的时机被提交或回滚。而通过SqlSession进行事务管理,虽然部分隐藏了底层细节,但为开发者提供了一个更为简单和直观的API,有助于简化代码并减少出错的机会
在Spring框架中,事务管理是通过AOP技术实现的,它提供了一种高效、灵活的方法来管理事务。事务管理在Spring中的实现与我们之前在AOP中讨论的基本概念紧密相连:原始对象、额外功能、切入点和组装切面。
DataSourceTransactionManager
类。这个类为我们处理了大部分与事务管理相关的复杂性,如事务的开启、提交和回滚等。@Transactional
注解来标识一个切入点。这告诉Spring哪些方法需要事务管理。当标注了此注解的方法被调用时,Spring会自动为其提供事务管理服务。
标签来启用基于注解的事务管理。此标签告诉Spring去查找带有@Transactional注解的方法,并为其提供事务管理服务。为了使这一切正常工作,我们还需要提供一个DataSourceTransactionManager的引用,它会负责实际的事务管理工作步骤如下
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-txartifactId>
<version>5.1.14.RELEASEversion>
dependency>
@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("测试异常");
}
}
<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">
<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>
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动态代理可能是首选,因为它通常有更好的性能。但是,如果业务对象没有实现任何接口或者有某些原因需要使用Cglib,那么就可以选择Cglib动态代理。
总之,事实上Spring通过proxy-target-class属性提供了对事务管理代理方式的控制,进一步证明了其事务管理是基于AOP的实现,无论是通过JDK动态代理还是Cglib动态代理,都是遵循AOP的核心原则和概念
<tx:annotation-driven transaction-manager="dataSourceTransactionManager" proxy-target-class="true"/>
在这篇《Spring5应用之事务处理》中,我们详细探讨了Spring中事务的关键概念和实现机制。首先,我们深入理解了事务的核心特性AICD,并通过对比JDBC和MyBatis在事务处理上的实现,进一步明确了事务控制的细节和特点。重要的是,我们发现Spring通过AOP技术巧妙地进行事务管理,从而大大简化了开发者的工作。
文章强调了原始对象如何与额外功能相结合,以及如何通过特定的注解和配置实现事务控制。特别是通过深入研究tx:annotation-driven标签,我们进一步了解了Spring在事务处理背后的AOP实现机制,如何选择不同的代理方法来达到我们的需求。
总而言之,本文为我们提供了一个全面、系统的视角,看待Spring在事务处理上的优越性和实现方法,帮助我们更加自信地在实际开发中应用Spring进行事务管理