事务概念

1、什么事务

(1)事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败则所有操作都失败
(2)典型场景:银行转账
lucy 转账 100 元 给 mary
lucy 少 100,mary 多 100

2、事务四个特性(ACID)

(1)原子性
(2)一致性
(3)隔离性
(4)持久性

事务操作(搭建事务操作环境)

图6.png

1、创建数据库表,添加记录

QQ截图20220422164532.png

2、创建 service,搭建 dao,完成对象创建和注入关系

(1)service 注入 dao,在 dao 注入 JdbcTemplate,在 JdbcTemplate 注入 DataSource

  1. @Service
  2. public class UserService {
  3. @Autowired
  4. private UserDao userDao;
  5. }
  6. @Repository
  7. public class UserDaoImpl implements UserDao {
  8. @Autowired
  9. private JdbcTemplate jdbcTemplate;
  10. }

3、在 dao 创建两个方法:多钱和少钱的方法,在 service 创建方法(转账的方法)

  1. @Repository
  2. public class UserDaoImpl implements UserDao {
  3. @Autowired
  4. private JdbcTemplate jdbcTemplate;
  5. @Override
  6. public void addMoney() {
  7. String sql = "update t_account set money=money+? where username=?";
  8. jdbcTemplate.update(sql,100,"mary");
  9. }
  10. @Override
  11. public void reduceMoney() {
  12. String sql = "update t_account set money=money-? where username=?";
  13. jdbcTemplate.update(sql,100,"lucy");
  14. }
  15. }
  1. @Service
  2. public class UserService {
  3. @Autowired
  4. private UserDao userDao;
  5. public void accountMoney() {
  6. userDao.reduceMoney();
  7. userDao.addMoney();
  8. }
  9. }

4、上面代码,如果正常执行没有问题的,但是如果代码执行过程中出现异常,有问题

  1. @Service
  2. public class UserService {
  3. @Autowired
  4. private UserDao userDao;
  5. public void accountMoney() {
  6. userDao.reduceMoney();
  7. //模拟异常
  8. int i = 10/0;
  9. userDao.addMoney();
  10. }
  11. }

(1)上面问题如何解决呢?
* 使用事务进行解决
(2)事务操作过程

  1. public void accountMoney() {
  2. try {
  3. //1、开启事物
  4. //2、进行业务操作
  5. userDao.reduceMoney();
  6. //模拟异常
  7. int i = 10/0;
  8. userDao.addMoney();
  9. //第三步没有发生异常,提交事务
  10. } catch (Exception e) {
  11. //第四步出现异常,事务回滚
  12. }
  13. }
  14. //此为编程式事物管理

事务操作(Spring 事务管理介绍)

1、事务添加到 JavaEE 三层结构里面 Service 层(业务逻辑层)
2、在 Spring 进行事务管理操作
(1)有两种方式:编程式事务管理和声明式事务管理(使用)
3、声明式事务管理
(1)基于注解方式(使用)
(2)基于 xml 配置文件方式
4、在 Spring 进行声明式事务管理,底层使用 AOP 原理
5、Spring 事务管理 API
(1)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类
QQ截图20220423091412.png

事务操作(注解声明式事务管理)

1、在 spring 配置文件配置事务管理器

  1. <!--创建事务管理器-->
  2. <bean id="transactionManager"
  3. class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  4. <!--注入数据源-->
  5. <property name="dataSource" ref="dataSource"></property>
  6. </bean>

2、在 spring 配置文件,开启事务注解

(1)在 spring 配置文件引入名称空间 tx

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:aop="http://www.springframework.org/schema/aop"
  6. xmlns:tx="http://www.springframework.org/schema/tx"
  7. xsi:schemaLocation="http://www.springframework.org/schema/beans
  8. http://www.springframework.org/schema/beans/spring-beans.xsd
  9. http://www.springframework.org/schema/context
  10. http://www.springframework.org/schema/context/spring-context.xsd
  11. http://www.springframework.org/schema/aop
  12. http://www.springframework.org/schema/aop/spring-aop.xsd
  13. http://www.springframework.org/schema/tx
  14. http://www.springframework.org/schema/tx/spring-tx.xsd
  15. ">

(2)开启事务注解

  1. <!--开启事务注解-->
  2. <!-- 开启基于注解的声明式事务功能 -->
  3. <!-- 使用transaction-manager属性指定当前使用是事务管理器的bean -->
  4. <!-- transaction-manager属性的默认值是transactionManager,如果事务管理器bean的id正好就是这个默认值,则可以省略这个属性 -->
  5. <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>

3、在 service 类上面(或者 service 类里面方法上面)添加事务注解

(1)@Transactional,这个注解添加到类上面,也可以添加方法上面
@Transactional 注解只能应用到 public 方法才有效
(2)如果把这个注解添加类上面,这个类里面所有的方法都添加事务
(3)如果把这个注解添加方法上面,为这个方法添加事务

  1. @Service
  2. @Transactional
  3. public class UserService {

事务操作(声明式事务管理参数配置)

1、在 service 类上面添加注解@Transactional,在这个注解里面可以配置事务相关参数

QQ截图20220423093421.png

2、propagation:事务传播行为

(1)多事务的方法之间进行调用,这个过程中事务 是如何进行管理的
图7.png
图上文字不重要

事务的传播行为可以由传播属性指定。Spring定义了7种类传播行为。

传播属性 描述
REQUIRED 如果有事务在运行,当前的方法就在这个事务内运行,否则,就启动一个新的事务,并在自己的事务内运行
REQUIRED_NEW 当前的方法必须启动新事务,并在它自己的事务内运行.如果有事务正在运行,应该将它挂起
SUPPORTS 如果有事务在运行,当前的方法就在这个事务内运行。否则它可以不运行在事务中.
NOT_SUPPORTED 当前的方法不应该运行在事务中,如果有运行的事务,将它挂起
MANDATORY 当前的方法必须运行在事务内部,如果没有正在运行的事务,就抛出异常
NEVER 当前的方法不应该运行在事务中。如果有运行的事务,就抛出异常
NESTED 如果有事务在运行,当前的方法就应该在这个事务的嵌套事务内运行。否则,就启动一-个新的事务,并在它自己的事务内运行。

嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。


PROPAGATION_NESTED 与PROPAGATION_REQUIRES_NEW的区别:

它们非常类似,都像一个嵌套事务,如果不存在一个活动的事务,都会开启一个新的事务。
使用 PROPAGATION_REQUIRES_NEW时,内层事务与外层事务就像两个独立的事务一样,一旦内层事务进行了提交后,外层事务不能对其进行回滚。两个事务互不影响。两个事务不是一个真正的嵌套事务。同时它需要JTA事务管理器的支持。
使用PROPAGATION_NESTED时,外层事务的回滚可以引起内层事务的回滚。而内层事务的异常并不会导致外层事务的回滚,它是一个真正的嵌套事务。DataSourceTransactionManager使用savepoint支持PROPAGATION_NESTED时,需要JDBC 3.0以上驱动及1.4以上的JDK版本支持。其它的JTATrasactionManager实现可能有不同的支持方式。
PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 “内部” 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行。
另一方面, PROPAGATION_NESTED 开始一个 “嵌套的” 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交。
由此可见, PROPAGATION_REQUIRES_NEW 和 PROPAGATION_NESTED 的最大区别在于, PROPAGATION_REQUIRES_NEW 完全是一个新的事务, 而 PROPAGATION_NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于 roll back.
参考:看完就明白_spring事务的7种传播行为


  1. @Service
  2. @Transactional(propagation = Propagation.REQUIRED)
  3. public class UserService {

3、ioslation:事务隔离级别

(1)事务有特性成为隔离性,多事务操作之间不会产生影响。不考虑隔离性产生很多问题
(2)有三个读问题:脏读、不可重复读、虚(幻)读
(3)脏读:一个未提交事务读取到另一个未提交事务的数据
图8.png
(4)不可重复读:一个未提交事务读取到另一提交事务修改数据
(5)虚读:一个未提交事务读取到另一提交事务添加数据
(6)解决:通过设置事务隔离级别,解决读问题

脏读 不可重复读 幻读
READ UNCOMMITTED
(读未提交)。
READ COMMITTED
(读已提交)。
REPEATABLE READ
(可重复读)
SERIALIZABLE
(串行化) 。
  1. @Service
  2. @Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.REPEATABLE_READ)
  3. public class UserService {

4、timeout:超时时间

(1)事务需要在一定时间内进行提交,如果不提交就进行回滚
(2)默认值是 -1 ,设置时间以秒单位进行计算 (-1是没有超时限制)

5、readOnly:是否只读

(1)读:查询操作,写:添加修改删除操作
(2)readOnly 默认值 false,表示可以查询,可以添加修改删除操作
(3)设置 readOnly 值是 true,设置成 true 之后,只能查询

6、rollbackFor:回滚

(1)设置出现哪些异常进行事务回滚

7、noRollbackFor:不回滚

(1)设置出现哪些异常不进行事务回滚

事务操作(XML 声明式事务管理)

1、在 spring 配置文件中进行配置
第一步 配置事务管理器
第二步 配置通知

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:aop="http://www.springframework.org/schema/aop"
  6. xmlns:tx="http://www.springframework.org/schema/tx"
  7. xsi:schemaLocation="http://www.springframework.org/schema/beans
  8. http://www.springframework.org/schema/beans/spring-beans.xsd
  9. http://www.springframework.org/schema/context
  10. http://www.springframework.org/schema/context/spring-context.xsd
  11. http://www.springframework.org/schema/aop
  12. http://www.springframework.org/schema/aop/spring-aop.xsd
  13. http://www.springframework.org/schema/tx
  14. http://www.springframework.org/schema/tx/spring-tx.xsd
  15. ">
  16. <!-- 组件扫描 -->
  17. <context:component-scan base-package="com.atguigu"></context:component-scan>
  18. <!-- 数据库连接池 -->
  19. <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
  20. destroy-method="close">
  21. <property name="url" value="jdbc:mysql://localhost:13306/user_db"/>
  22. <property name="username" value="root"/>
  23. <property name="password" value="742153"/>
  24. <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
  25. </bean>
  26. <!-- JdbcTemplate 对象 -->
  27. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
  28. <!--注入 dataSource-->
  29. <property name="dataSource" ref="dataSource"></property>
  30. </bean>
  31. <!--1 创建事务管理器-->
  32. <bean id="transactionManager"
  33. class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  34. <!--注入数据源-->
  35. <property name="dataSource" ref="dataSource"></property>
  36. </bean>
  37. <!--2 配置通知-->
  38. <tx:advice id="txadvice" transaction-manager="transactionManager">
  39. <!--配置事务参数-->
  40. <tx:attributes>
  41. <!--指定哪种规则的方法上面添加事务-->
  42. <tx:method name="accountMoney" propagation="REQUIRED"/>
  43. <!--<tx:method name="account*"/>-->
  44. </tx:attributes>
  45. </tx:advice>
  46. <!--3 配置切入点和切面-->
  47. <aop:config>
  48. <!--配置切入点-->
  49. <aop:pointcut id="pt" expression="execution(* com.atguigu.spring5.service.UserService.*(..))"/>
  50. <!--配置切面-->
  51. <aop:advisor advice-ref="txadvice" pointcut-ref="pt"/>
  52. </aop:config>
  53. </beans>

事务操作(完全注解声明式事务管理)

1、创建配置类,使用配置类替代 xml 配置文件

  1. @Configuration //配置类
  2. @ComponentScan(basePackages = "com.atguigu") //组件扫描
  3. @EnableTransactionManagement //开启事务
  4. public class TxConfig {
  5. //创建数据库连接池
  6. @Bean
  7. public DruidDataSource getDruidDataSource() {
  8. DruidDataSource dataSource = new DruidDataSource();
  9. dataSource.setDriverClassName("com.mysql.jdbc.Driver");
  10. dataSource.setUrl("jdbc:mysql://localhost:13306/user_db");
  11. dataSource.setUsername("root");
  12. dataSource.setPassword("742153");
  13. return dataSource;
  14. }
  15. //创建 JdbcTemplate 对象
  16. @Bean
  17. public JdbcTemplate getJdbcTemplate(DataSource dataSource) {
  18. //到 ioc 容器中根据类型找到 dataSource
  19. JdbcTemplate jdbcTemplate = new JdbcTemplate();
  20. //注入 dataSource
  21. jdbcTemplate.setDataSource(dataSource);
  22. return jdbcTemplate;
  23. }
  24. //创建事务管理器
  25. @Bean
  26. public DataSourceTransactionManager
  27. getDataSourceTransactionManager(DataSource dataSource) {
  28. DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
  29. transactionManager.setDataSource(dataSource);
  30. return transactionManager;
  31. }
  32. }

@Bean明确地指示了一种方法,什么方法呢——产生一个bean的方法,并且交给Spring容器管理;从这我们就明白了为啥@Bean是放在方法的注释上了,因为它很明确地告诉被注释的方法,你给我产生一个Bean,然后交给Spring容器,剩下的你就别管了。


@Configuration标注在类上,相当于把该类作为spring的xml配置文件中的,作用为:配置spring容器(应用上下文)

@Bean标注在方法上(返回某个实例的方法),等价于spring的xml配置文件中的,目的是创建一个类,作用为:当spring需要创建指定的一个类时会调用这个注解(@Bean)的方法。
注:
(1)、@Bean注解在返回实例的方法上,如果未通过@Bean指定bean的名称,则默认与标注的方法名相同;
(2)、@Bean注解默认作用域为单例singleton作用域,可通过@Scope(“prototype”)设置为原型作用域;
(3)、既然@Bean的作用是注册bean对象,那么完全可以使用@Component、@Controller、@Service、@Ripository等注解注册bean,当然需要配置@ComponentScan注解进行自动扫描。

@Configuation等价于
@Bean等价于
@ComponentScan等价于
————————————————
版权声明:本文为CSDN博主「liuyinfei_java」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/liuyinfei_java/article/details/82011805


@ComponentScan(value = “XXX”) 是用来告诉spring去哪扫描要注入的bean。为了兼容及灵活配置扫描路径,这个注解定义了很多的参数,具体的:
basePackages与value: 用于指定包的路径,进行扫描

basePackageClasses: 用于指定某个类的包的路径进行扫描
nameGenerator: bean的名称的生成器
useDefaultFilters: 是否开启对@Component,@Repository,@Service,@Controller的类进行检测
includeFilters: 包含的过滤条件
FilterType.ANNOTATION:按照注解过滤
FilterType.ASSIGNABLE_TYPE:按照给定的类型
FilterType.ASPECTJ:使用ASPECTJ表达式 (不常用)
FilterType.REGEX:正则 (不常用)
FilterType.CUSTOM:自定义规则
excludeFilters: 排除的过滤条件,用法和includeFilters一样

  1. @Controller、@Service、@Component、@Repository注解的类都会被扫描成bean,注册到容器
  2. 但接口类型即使加上@Component等注解,也不会实例化成bean,比如这里的 DeptDao 接口,可以看到并未生成对应的bean。

————————————————
版权声明:本文为CSDN博主「沙滩de流沙」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_41231928/article/details/116504236