【棘手问题】Spring JPA一级缓存导致获取不到数据库表中的最新数据,对象地址不发生改变

【棘手问题】Spring JPA一级缓存导致获取不到数据库表中的最新数据,对象地址不发生改变

  • 一、问题背景
  • 二、解决步骤
    • 2.1 debug
    • 2.2 原因分析
      • 2.2.1 数据步骤
      • 2.2.2 大模型解释
      • 2.2.3 解释举例
      • 2.2.4 关键函数
    • 2.3 解决方案
  • 三、Spring JPA一级缓存

一、问题背景

项目的数据可以通过前端表单进行增删改查,亦可以通过Excel表格上传的方式进行项目数据增改。两种方式对数据库数据进行操作。

项目可以通过字段 项目ID项目标识字段均能够查找到唯一项目记录。

问题发生于:
修改项目的负责人数据时,与其关联的资产数据的负责人要同步跟随更新。就在判断来自Excel表中的项目负责人数据库表中的负责人是否一致时,出现了异常,不一致的分支条件始终没进入。

二、解决步骤

2.1 debug

这种问题,必须debug看其所调用的函数内部对象数据的变化。

首先看一段代码,核心函数是createOrUpdateAssetProjectByExcel(assetProject),功能顾名思义。

 public ResponseEntity<Object> importExcel(List<AssetProjectDto> assetProjectDtoList, boolean updateSupport) {if (CollectionUtils.isEmpty(assetProjectDtoList)){throw new BadRequestException("导入的表格为空");}try {assetProjectDtoList.stream()
//                    .filter(assetProjectDto -> ObjectUtil.isNotNull(assetProjectDto.getProjectNumber())).filter(assetProjectDto -> ObjectUtil.isNotNull(assetProjectDto.getUserId())).map(assetProjectDto -> BeanUtil.toBeanIgnoreError( assetProjectDto, AssetProject.class)).forEach((assetProject -> {try{createOrUpdateAssetProjectByExcel(assetProject);}catch (Exception e){log.error("项目创建或更新失败,原因是[{}]",ThrowableUtil.getMessage(e));}}));}catch (Exception e){throw new BadRequestException("导入失败,原因是:" + e.getMessage());}return new ResponseEntity<>("导入成功", HttpStatus.OK);}

接着看一下该函数,通过判断项目是否是新项目对该条数据进行更新或者创建的操作。
在这里插入图片描述
进入到更新函数中,就出现了resourceassetProject对象始终是相等的问题。
在这里插入图片描述

2.2 原因分析

2.2.1 数据步骤

排除了事务、属性复制的原因,最后发现问题出现在resourceassetProject对象都是从数据库中查询出来的,而且经过属性拷贝。

  1. resource对象通过项目标识查询出来;
  2. excel数据对象的属性复制给resource对象;
  3. assetProject对象通过项目ID查询出赖;
  4. assetProject对象和resource对象一致。

在第3步中,查询出的项目对象走了Spring JPA一级缓存导致,第4步的结果。

2.2.2 大模型解释

在 JPA 中,findByProjectItem 函数是一个自定义的查询方法,它根据实体的某个属性(在这个例子中是 projectItem)来检索实体。这个方法的名字是任意的,但它后面的参数定义了查询的条件。
findById 是一个标准的方法,它根据实体的主键(ID)来检索实体。这是一个 JPA 定义的通用查询方法,它总是根据 ID 来查询。
如果 findByProjectItem 查询返回的实体 ID 和 findById 查询返回的实体 ID 相同,这意味着根据 projectItem 属性的查询条件找到了一个实体,这个实体的 ID 恰好也是 findById 查询所指定的 ID。
在这种情况下,由于 JPA 的唯一性保证,这两个查询将会返回同一个实体实例。这是因为 JPA 确保了基于主键的查询总是返回唯一的实体实例,而且主键是实体的唯一标识符。因此,如果两个查询都指向同一个实体 ID,那么它们将会返回内存中地址相同的那个实体对象。

这里有一个重要的点:
即使两个查询的条件不同,如果它们都返回同一个实体 ID,那么 JPA 会认为它们是相同的实体,并且返回同一个对象实例。这是 JPA 规范的行为,也是 Hibernate(JPA 的一个实现)的行为。
总结来说,如果 findByProjectItem 查询和 findById 查询返回相同的实体 ID,那么它们将会返回同一个对象实例。这是因为 JPA 规范要求实体 ID 是唯一的,并且基于 ID 的查询总是返回唯一的实体。

在 JPA 中,实体的唯一性是通过实体类的 @Id 注解定义的主键来实现的。每个实体类都必须有一个或多个 @Id 注解的属性,这些属性共同构成了实体的主键。主键的值是唯一的,JPA 使用这个值来识别和检索实体。
在 Hibernate(JPA 的一个流行实现)中,这种唯一性是通过 EntityManagerfind 方法和 getReference 方法来保证的。当您使用这些方法根据主键查询实体时,Hibernate 会检查主键是否已经存在于当前的持久化上下文中。如果主键存在,它将返回已经存在的实体;如果不存在,它将创建一个新的实体实例并将其持久化。

2.2.3 解释举例

以下是一个简单的示例,在 JPA 实体类中定义主键:

@Entity
public class AssetProject {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;// 其他属性public Long getId() {return id;}// 构造器、getter 和 setter
}

在这个例子中,id 属性被定义为实体的主键,并且使用了 @GeneratedValue 注解来指定主键值的生成策略。通常,当您使用 JPA 提供的 EntityManager 进行持久化操作时,JPA 会处理实体的唯一性检查。
在 Spring Data JPA 中,通常不需要直接处理这些底层细节。Spring Data JPA 会为提供简化的接口和方法,如 findById,它会隐式地处理实体的唯一性检查。当调用 findById 方法并提供一个 ID 时,Spring Data JPA 会确保返回的实体是唯一的,并且与提供的 ID 相匹配。
如果您想深入了解 JPA 和 Hibernate 是如何实现这种唯一性的,您可以查看它们的源代码。JPA 的核心规范可以在 EclipseLink、Hibernate、TopLink 等实现中找到。这些实现中的 EntityManager 类和相关的持久化方法负责处理实体的唯一性检查和检索。

2.2.4 关键函数

2.3 解决方案

在查询前,清除缓存。
在这里插入图片描述

三、Spring JPA一级缓存

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/731786.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

STM32 通过Modbus协议更改内部Flash(模仿EEPROM)的运行参数

main.c测试 uint8_t uart1RxBuf[64]{0};uint8_t Adc1ConvEnd0; uint8_t Adc2ConvEnd0;int main(void) {/* USER CODE BEGIN 1 *//* USER CODE END 1 *//* MCU Configuration--------------------------------------------------------*//* Reset of all peripherals, Initial…

C++ 智能指针深度剖析

文章目录 1. 前言2. 为什么需要智能指针&#xff1f;3. 内存泄漏3.1 内存泄漏的概念及危害3.2 内存泄漏的分类3.3 如何检测内存泄漏3.4 如何避免内存泄漏 4. 智能指针的使用及原理4.1 RAII思想4.2 智能指针的原理4.3 C智能指针发展历史4.4 std::auto_ptr4.5 std::unique_ptr4.6…

掌握MySQL,看完这篇文章就够了!

1. MySQL MySQL是一种广泛使用的关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;由瑞典的MySQL AB公司开发&#xff0c;目前属于甲骨文公司&#xff08;Oracle Corporation&#xff09;。 MySQL使用结构化查询语言&#xff08;SQL&#xff09;进行数据库管理…

C++的面向诗篇:类的叙事与对象的旋律

个人主页&#xff1a;日刷百题 系列专栏&#xff1a;〖C/C小游戏〗〖Linux〗〖数据结构〗 〖C语言〗 &#x1f30e;欢迎各位→点赞&#x1f44d;收藏⭐️留言&#x1f4dd; ​ ​ 一、面向对象的定义 学习C语言时&#xff0c;我们就经常听说C语言是面向过程的&#xff0c;…

[nlp入门论文精读] | Transformer

写在前面 最近工作从CV转向了NLP&#xff0c;于是空余时间便跟着哔哩哔哩李沐老师的视频学习。其实研一NLP课程讲论文的时候&#xff0c;我们小组就选择了经典的Attention和Bert&#xff0c;但还有很多细节并不完全理解&#xff0c;实际使用时也很困惑。 因此这个系列就来记…

红队专题-开源漏扫-巡风xunfeng源码剖析与应用

开源漏扫-巡风xunfeng 介绍主体两部分:网络资产识别引擎,漏洞检测引擎。代码赏析插件编写JSON标示符Python脚本此外系统内嵌了辅助验证功能文件结构功能 模块添加IP三. 进行扫描在这里插入图片描述 ![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/de587a6f6f694…

ai写作一键生成文章速度快

ai写作是一种基于人工智能技术的自动写作工具&#xff0c;它可以根据用户提供的主题或关键词&#xff0c;快速一键生成文章。与传统的手动写作相比&#xff0c;ai写作的速度更快&#xff0c;效率更高。下面小编就带大家一起来见识下ai写作一键生成文章的速度是如何之快&#xf…

【CSP试题回顾】202012-1-期末预测之安全指数

CSP-202012-1-期末预测之安全指数 解题代码 #include <iostream> #include <algorithm> using namespace std;int n, sum;int main() {cin >> n;for (int i 0; i < n; i){int w, s;cin >> w >> s;sum w * s;}sum max(sum, 0);cout <&…

Electron-builder打包安装包——编译篇

突然有一天想打包个桌面程序&#xff0c;没有打包过&#xff0c;经过九牛二虎之力终于打包出来&#xff0c;在此感谢那些热于分享的前辈&#xff01; 本篇只讲打包运行和出现的问题 一、准备工作&#xff1a;提前下载相关资源包&#xff0c;否则在国内环境下可能因为网络问题…

ECharts 简要介绍及简单实例代码

ECharts 是一个使用 JavaScript 实现的开源可视化库&#xff0c;涵盖各行业图表&#xff0c;满足各种需求。 ECharts 提供了丰富的图表类型和交互能力&#xff0c;使用户能够通过简单的配置生成各种各样的图表&#xff0c;包括但不限于折线图、柱状图、散点图、饼图、雷达图、…

Lwip之TCP服务端示例记录(1对多)

前言 实现多个客户端同时连接初步代码结构已经实现完成(通过轮训的方式) // // Created by shchl on 2024/3/8. // #if 1#include <string.h> #include "lwip/api.h" #include "FreeRTOS.h" #include "task.h" #include "usart.h&…

群辉docker安装sql server

安装步骤 开启群辉 SSH&#xff0c;通过 SSH 工具连接到群辉&#xff0c;运行下面的命令拉取mssql 2019 镜像 sudo docker pull mcr.microsoft.com/mssql/server:2019-latest然后在 docker 中就可以看到该镜像&#xff1a; 在群晖 docker 共享文件夹中创建 mssql2009 文件夹 …

Qt插件之输入法插件的构建和使用(二)

文章目录 主键盘搭建Google开源引擎音节分割工具类参考项目下载搭建好各个基础控件之后,就可以开发输入法的主界面和引擎了,这也是输入法的核心。 主键盘搭建 输入法的主界面本质上是一个QStackedWidget容器,将各个类型的输入键盘插入到容器中,然后根据业务需要切换不同的…

金现代产品方案部部长王宁,将出席“ISIG-低代码/零代码技术与应用发展峰会”

3月16日&#xff0c;第四届「ISIG中国产业智能大会」将在上海中庚聚龙酒店拉开序幕。本届大会由苏州市金融科技协会指导&#xff0c;企智未来科技&#xff08;LowCode低码时代、RPA中国、AIGC开放社区&#xff09;主办。大会旨在聚合每一位产业成员的力量&#xff0c;深入探索低…

生产管理MES系统在卫浴企业中的应用

在卫浴企业的制造过程中&#xff0c;实现透明车间管理是一个旨在提升效率、质量和可视性的重要目标。MES系统可以在卫浴企业中帮助打造透明车间&#xff0c;提升生产管理的效率和可视性。具体能实现哪些管理呢&#xff1f; 实时监控生产状态&#xff1a; MES系统可以实时监控车…

【C#杂谈】在 .NET Framework 中使用新的C#语言特性

前排提示&#xff1a;提出一个可以让 [^1] 这中语法可以在.NET Framework运行时中使用的方法 众所都周知&#xff0c;.NET Framework&#xff08;以下简称 .NF&#xff09;作为一个被微软官方确认不在继续发布新特性的运行时&#xff0c;它所对应的C#语言版本被&#xff08;官方…

RabbitMQ架构详解

文章目录 概述架构详解核心组件虚拟主机&#xff08;Virtual Host&#xff09;RabbitMQ 有几种广播类型 概述 RabbitMQ是⼀个高可用的消息中间件&#xff0c;支持多种协议和集群扩展。并且支持消息持久化和镜像队列&#xff0c;适用于对消息可靠性较高的场合 官网https://www.…

uniapp富文本编辑-editor-vue2-vue3-wangeditor

前言 除了“微信小程序”&#xff0c;其他小程序想要使用editor组件实现富文本编辑&#xff0c;很难vue3项目 官方组件editor&#xff0c;在初始化时有点麻烦&#xff0c;建议搭配第三方组件wangeditor 写在前面 - editor组件缺少editor-icon.css 内容另存为editor-icon.css…

服务器租用和托管的区别

目前对于服务器要求相对高的企业会希望使用独立服务器来运行自己的网站&#xff0c;而在选择独立服务器业务事&#xff0c;是使用服务器托管还是服务器租用这两种方法时&#xff0c;许多刚进入网络或者传统行业的从业者&#xff0c;都不太了解什么是服务器&#xff0c;现在我来…

系统安全保证措施-word

【系统安全保证措施-各支撑材料直接套用】 一、 身份鉴别 二、 访问控制 三、 通信完整性、保密性 四、 抗抵赖 五、 数据完整性 六、 数据保密性 七、 应用安全支撑系统设计 软件全套资料下载进主页。