SpringBoot中@Transiactional注解没有效果的解决

目录
  • SpringBoot @Transiactional注解没有效果
    • 背景
    • 问题
    • 解决
  • SpringBoot 使用Transaction注解遇到的坑
    • 一、场景
    • 二、Spring中使用的使用方式
    • 三、使用中遇到的问题
      • 1、使用Transaction注解时抛出异常但是事务不起作用,异常时事务没有进行回滚?
      • 2、刚插入的数据,无法马上查询到?
  • 总结:

SpringBoot @Transiactional注解没有效果

背景

数据库为mysql

问题

使用SpringBoot操作数据库插入两条数据,service层的方法出现了异常,按理说两条数据都该插不进去的,可以数据库中却还是有一条数据。

数据库表格式:

service层代码:

package com.example.demo.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import com.example.demo.domain.Girl;
import com.example.demo.repository.GirlRepository;
@Service
public class GirlService {
    @Autowired
    private GirlRepository girlRepository;
    @Transactional
    public void transiactionTest() {
        Girl g1 = new Girl();
        g1.setAge(33);
        g1.setCupSize("B");
        girlRepository.save(g1);
        Girl g2 = new Girl();
        g2.setAge(33);
        g2.setCupSize("FFFFF");  // 长度和数据库的长度不符,会出异常
        girlRepository.save(g2);
    }
}

启动SpringBoot后,访问对应的方法,控制台也报错,但是表中是有一条数据的(原本是空表)。

查了查资料说的是在设计表的时候要选取的InnoDB引擎

回头看我的表引擎:

还真的是。。。。。。

解决

将数据表的引擎设置为InnoDB引擎。 然后再次执行,@Transitional注解才起了作用,数据表中没有了数据。

SpringBoot 使用Transaction注解遇到的坑

一、场景

开发一个多批次入库的功能,功能中涉及到多个表间的操作,对数据库表的操作要么同时成功,要么同时失败,不然就会存在脏数据,所以使用到了事务这个知识点()。

划重点:重要的都使用红色标出来了,大家如果不想看我废话,直接跳到红色字体即可0...0

二、Spring中使用的使用方式

1、使用传统的手动开始,手动提交事务即:beginTransaction()、commit()、rollback()等事务管理相关的方法,这就是编程式事务管理。

2、使用Transaction注解的声明式事务,将事务的开启和提交交给Spring容器完成,这个也是本次我使用的方式,简单,但是使用时需要注意很多细节。

3、基于Spring AOP的切面的事务配置(本人很少使用这个,所以本文不重点刨析该知识点,想了解的可以到其他博客进行查看)

三、使用中遇到的问题

1、使用Transaction注解时抛出异常但是事务不起作用,异常时事务没有进行回滚?

答:经过排查,查询在开启事务的方法中最外层使用了try...catch进行了异常的捕获,因此抛出的异常本捕获了,切面无法捕获到异常,所以不会进行回滚。

解决:

(1) 手动指定切面捕获的异常类型(因为默认情况下只会在RuntimeExceptionimeException情况下才会进行事务的回滚),方式:@Transaction(rollbcackFor=Exception.class)

(2) 在catch中手动抛出一个运行时异常即:throw new RuntimeException();

(3) 如果需要在事务回滚时,给调用当前方法的调用者返回错误信息的话,用第二种方案就是不行的,因为抛出异常后的语句时不会执行的,包括return后面的语句,所以,此时可以手动进行事务回滚的语句调用即:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();

2、刚插入的数据,无法马上查询到?

答: 这个问题是个大坑,整整苦恼了我两天(可能是之前使用事务相关的知识比较少,所以遇到了根本就不知道是什么原因),因为项目中使用的是MybatisPlus框架,一开始便怀疑是框架的缓存问题,因为刚插入马上查询时,查询的SQL语句并没有执行,即根本没到数据库去查,但是,手动关闭了缓存之后并没起效果,最后大佬看了代码后,一眼就指出问题所在(现在不得不感慨,经验时多么重要)。

要解决这个问题,首先要知道使用Transaction是怎么进行事务增强的,说白了,是通过生成代理对象进行切面注入的,当前对象并没有增强的作用,刚开始我插入和查询的方法都是写在一个service中,然后使用this调用这些方法,而this表示的是当前的service对象,所以这些方法根本就不在当前的事务中,因为刚插入的数据无法马上查询到0....0(我踩的大坑希望大家不要再踩了,太难受了..)

解决方法:说了这么多废话,现在知道问题产生的原因,所以就好解决了:

(1) 将所有的数据库操作方法抽取到另外一个Service对象中,然后通过@Autowire注入调用即可。

(2) 自己注入自己的对象即当前Service为A,可以直接使用:@Autowire private A a; 然后通过a调用相关数据库操作的方法(注意不要使用this,使用this的话无效),@Autowire private A a这句话实际上返回的是当前Service的代理对象,但特别需要注意的是:所有操作数据库相关的方法,访问权限都需改成public,不然会出现mapper和service注入为null---这个是个大坑,具体原因还不知道(这个方式推荐使用)

(3) 使用:((A) AopContext.currentProxy()).方法名()进行调用(听说这个方式在打包发布的时候会出现问题,本次没有试过这个方式,所以不推荐)

总结:

经验就是一个积累的过程,没有谁能够一步登天,所以脚踏实地才是成功的秘诀。本文只是本人在使用事务知识是遇到的问题的总结,希望能给大家一个参考,也希望大家多多支持我们!

(0)

相关推荐

  • 解决springboot中@DynamicUpdate注解无效的问题

    springboot 中 @DynamicUpdate 注解无效解决方案 遇到的问题 项目中使用 jpa,以前没用过,所以踩坑在所难免. 在使用过程中,要更新一条记录的某个字段,更新成功以后,发现整条记录只剩下我更新的那个字段,其他的全部为空了. 瞬间明白,这种更新是全覆盖,针对每个字段 update,实体类没赋值的字段,也直接将空值 set 过去了. 寻求解决方案 正在庆幸这么容易就解决,突然发现并没有这么简单. 群众的力量是无穷大的,我立刻就明白这个注解为什么无效,原来是搞错了它的用途. 一

  • springboot中使用@Transactional注解事物不生效的坑

    一:在springboot中使用事物遇到的坑 1.我们知道spring中的事物分为两种:一种是编程式事物,一种是声明式事物.顾名思义,编程式事物是指通过代码去实现事物管理,这里不做过多说明.另一种是声明式事物,分为两种情况01:一种是通过传统xml方式配置,02:使用@Transaction注解方式配置,这是主要讲解的是通过注解方式配置.因为在springboot项目中,会自动配置DataSourceTransactionManager,我们只需要在对应的方法上或者类上加上@Transactio

  • 详解在SpringBoot中@Transactional事物操作和事物无效问题排查

    目录 1.spring事务管理简述 2.SpringBoot中使用@Transactional注解 2.1.开启事务注解 2.2.在目标类.方法上添加注解@Transactional 2.3.细化事务配置 3.@Transactional事务实现机制 3.1.整体事务控制流程 3.2.Spring AOP的两种代理 3.3.事务操作的底层实现 4.@Transactional使用注释实现及问题排查 4.1.数据库引擎是否支持事务? 4.3.注解所在的类是否被加载成Bean? 4.2.注解所在方法

  • 解决SpringBoot中使用@Async注解失效的问题

    错误示例,同一个类中使用异步方法: package com.xqnode.learning.controller; import com.fasterxml.jackson.core.JsonProcessingException; import org.springframework.scheduling.annotation.Async; import org.springframework.web.bind.annotation.GetMapping; import org.springf

  • SpringBoot中@Transiactional注解没有效果的解决

    目录 SpringBoot @Transiactional注解没有效果 背景 问题 解决 SpringBoot 使用Transaction注解遇到的坑 一.场景 二.Spring中使用的使用方式 三.使用中遇到的问题 1.使用Transaction注解时抛出异常但是事务不起作用,异常时事务没有进行回滚? 2.刚插入的数据,无法马上查询到? 总结: SpringBoot @Transiactional注解没有效果 背景 数据库为mysql 问题 使用SpringBoot操作数据库插入两条数据,se

  • SpringBoot中@ConfigurationProperties注解实现配置绑定的三种方法

    properties配置文件如下: human.name=Mr.Yu human.age=21 human.gender=male 如何把properties里面的配置绑定到JavaBean里面,以前我们的做法如下: public class PropertiesUtil { public static void getProperties(Person person) throws IOException { Properties properties = new Properties();

  • SpringBoot 中常用注解及各种注解作用

    本篇文章将介绍几种SpringBoot 中常用注解 其中,各注解的作用为: @PathVaribale 获取url中的数据 @RequestParam 获取请求参数的值 @GetMapping 组合注解,是@RequestMapping(method = RequestMethod.GET)的缩写 @RestController是@ResponseBody和@Controller的组合注解. @PathVaribale 获取url中的数据 看一个例子,如果我们需要获取Url=localhost:

  • 在springboot中使用注解将值注入参数的操作

    后端的许多管理系统需要登陆者的信息,如shiro登陆后,会将登陆者的信息存储在shiro的session,在使用时需要多行代码获取用户信息.可以把获取在shiro中的登陆者信息封装在一个类中,使用时获取.本文主要讲述如何使用注解将值注入参数,shiro的配置请自行百度. 定义注解 新建一个InfoAnnotation.java的注解类,用于注解参数,代码如下: @Target(ElementType.PARAMETER) @Retention(RetentionPolicy.RUNTIME) p

  • SpringBoot中自定义注解实现参数非空校验的示例

    前言 由于刚写项目不久,在写 web 后台接口时,经常会对前端传入的参数进行一些规则校验,如果入参较少还好,一旦需要校验的参数比较多,那么使用 if 校验会带来大量的重复性工作,并且代码看起来会非常冗余,所以我首先想到能否通过一些手段改进这点,让 Controller 层减少参数校验的冗余代码,提升代码的可阅读性. 经过阅读他人的代码,发现使用 annotation 注解是一个比较方便的手段,SpringBoot 自带的 @RequestParam 注解只会校验请求中该参数是否存在,但是该参数是

  • SpringBoot中Mybatis注解一对多和多对多查询实现示例

    目录 一.模拟的业务查询 二.对应的实体类如下 三.对应的建表语句和模拟数据如下 四.@One一对一映射 五.@Many一对多查询 六.@One @Many的总结 一.模拟的业务查询 系统中的用户user都有唯一对应的地址信息address,每个用户可以有多量车car,类似如下结构 |-- user |-- address |-- carList |-- car1 |-- car2 二.对应的实体类如下 @Data public class AddressPO { private Long id

  • SpringBoot中@Import注解的使用方式

    目录 一. @Import引入普通类 二. @Import引入配置类(@Configuration修饰的类) 三 .@Import引入ImportSelector的实现类 3.1 静态import场景(注入已知的类) 3.2 动态import场景(注入指定条件的类) 四. @Import引入ImportBeanDefinitionRegistrar的实现类 前言: @Import注解用来帮助我们把一些需要定义为Bean的类导入到IOC容器里面.下面我们就对@Import注解的使用做一个简单的总结

  • SpringBoot中@Import注解如何正确使用

    目录 简介 一.功能简介 二.示例 1.引入普通类 2.引入ImportSelector的实现类 (1)静态import场景(注入已知的类) (2)动态import场景(注入指定条件的类) 3.引入ImportBeanDefinitionRegister的实现类 简介 由于最近的项目需求,需要在把配置类导入到容器中,通过查询,使用@Import注解就能实现这个功能,@Import注解能够帮我们吧普通配置类(定义为Bean的类)导入到IOC容器中.该注解我们也能过在源码中经常看到,是框架层实现的重

  • 详解SpringBoot中@ConditionalOnClass注解的使用

    目录 一.@ConditionalOnClass注解初始 二.@ConditionalOnClass注解用法 1.使用value属性 2.使用name属性 三.@ConditionalOnClass是怎么实现的 四.总结 今天给大家带来的是springboot中的@ConditionalOnClass注解的用法.上次的@ConditionalOnBean注解还记得吗? 一.@ConditionalOnClass注解初始 看下@CodidtionalOnClass注解的定义, 需要注意的有两点,

随机推荐