Spring Boot JPA save之怪异

Spring Boot JPA save方法之怪异

项目场景

工作中的一个小需求,由于这个项目第一阶段只是引入,作为触发程序在后台跑数据,将下游的数据引入即可。所以不需要暴露在页面上。所以报错的话,是没有任何人感知的。所以需要提供一个监控程序,就是在遇到异常就记录下来,每日满足发生5次异常就发送邮件通知,通知后就不再通知。


问题描述

程序其实很简单,但是测试Demo的时候,由于使用的是JPA,使用的save方法保存,方法内部处理就是先判断isNew,然后去做update或者insert操作。问题是这个方法遇到数据库异常,比如主键冲突,虽然打印日志,但是异常实际上是被某个异常处理器处理掉了,根本报不出来。

java 复制代码
try{
    dao.save(entity);
}catch (Exception e) {
    log.error("error happened", e);
    monitorExceptionService.recordError(e);
}

这个错误日志是进入不到catch里面的。


问题原因

网上资料介绍要使用saveAndFlush方法,查看内部源码,其实就是多了一个flush方法。flush其实就是将sql预编译执行的结果发送给数据库得到的结果。save方法是在sql统一提交的才会报错,就是@Transactional整体结束的时候报错,所以catch不到报错。而saveAndFlush方法多的flush方法就是将sql结果发送到数据库,所以立马能得到报错。

源码如下:

java 复制代码
	@Transactional
	@Override
	public <S extends T> S saveAndFlush(S entity) {

		S result = save(entity);
		flush();

		return result;
	}

解决方案

就是将save方法改为saveAndFlush方法即可。

代码如下:

java 复制代码
try{
    dao.saveAndFlush(entity);
}catch (Exception e) {
    log.error("error happened", e);
    monitorExceptionService.recordError(e);
}

搞定收工!

相关推荐
RainbowSea1 小时前
问题:后端由于字符内容过长,前端展示精度丢失修复
java·spring boot·后端
风象南2 小时前
SpringBoot 控制器的动态注册与卸载
java·spring boot·后端
我是一只代码狗2 小时前
springboot中使用线程池
java·spring boot·后端
hello早上好2 小时前
JDK 代理原理
java·spring boot·spring
PanZonghui2 小时前
Centos项目部署之运行SpringBoot打包后的jar文件
linux·spring boot
沉着的码农3 小时前
【设计模式】基于责任链模式的参数校验
java·spring boot·分布式
zyxzyx6663 小时前
Flyway 介绍以及与 Spring Boot 集成指南
spring boot·笔记
一头生产的驴5 小时前
java整合itext pdf实现自定义PDF文件格式导出
java·spring boot·pdf·itextpdf
程序员张37 小时前
SpringBoot计时一次请求耗时
java·spring boot·后端
麦兜*14 小时前
Spring Boot启动优化7板斧(延迟初始化、组件扫描精准打击、JVM参数调优):砍掉70%启动时间的魔鬼实践
java·jvm·spring boot·后端·spring·spring cloud·系统架构