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

本文主要是介绍【棘手问题】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一级缓存

这篇关于【棘手问题】Spring JPA一级缓存导致获取不到数据库表中的最新数据,对象地址不发生改变的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/789330

相关文章

Spring @Scheduled注解及工作原理

《Spring@Scheduled注解及工作原理》Spring的@Scheduled注解用于标记定时任务,无需额外库,需配置@EnableScheduling,设置fixedRate、fixedDe... 目录1.@Scheduled注解定义2.配置 @Scheduled2.1 开启定时任务支持2.2 创建

PostgreSQL数据库密码被遗忘时的操作步骤

《PostgreSQL数据库密码被遗忘时的操作步骤》密码遗忘是常见的用户问题,因此提供一种安全的遗忘密码找回机制是十分必要的,:本文主要介绍PostgreSQL数据库密码被遗忘时的操作步骤的相关资... 目录前言一、背景知识二、Windows环境下的解决步骤1. 找到PostgreSQL安装目录2. 修改p

SpringBoot中使用Flux实现流式返回的方法小结

《SpringBoot中使用Flux实现流式返回的方法小结》文章介绍流式返回(StreamingResponse)在SpringBoot中通过Flux实现,优势包括提升用户体验、降低内存消耗、支持长连... 目录背景流式返回的核心概念与优势1. 提升用户体验2. 降低内存消耗3. 支持长连接与实时通信在Sp

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

javax.net.ssl.SSLHandshakeException:异常原因及解决方案

《javax.net.ssl.SSLHandshakeException:异常原因及解决方案》javax.net.ssl.SSLHandshakeException是一个SSL握手异常,通常在建立SS... 目录报错原因在程序中绕过服务器的安全验证注意点最后多说一句报错原因一般出现这种问题是因为目标服务器

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使