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

news2024/10/7 18:27:56

【棘手问题】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.coloradmin.cn/o/1501854.html

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

相关文章

在ubuntu上使用vscode+gcc-arm-none-eabi+openocd工具开发STM32

文章目录 所需工具安装调试搭建过程中遇到的问题 写在前面 老大上周让我用vscode开发STM32&#xff0c;我爽快的答应了&#xff0c;心想大学四年装了这么多环境了这不简简单单&#xff0c;更何况vscode这两年还用过&#xff0c;然而现实总是令人不快的——我竟然花了差不多两周…

Java SE入门及基础(29)

第三节 访问修饰符 1. 概念 访问修饰符就是控制访问权限的修饰符号 2. 类的访问修饰符 类的访问修饰符只有两种&#xff1a;public 修饰符和默认修饰符&#xff08;不写修饰符就是默认&#xff09; public 修饰符修饰类表示类可以公开访问。默认修饰符修饰类表示该类只能…

flutter逆向 ACTF native ap

言 算了一下好长时间没打过CTF了,前两天看到ACTF逆向有道flutter逆向题就过来玩玩啦,花了一个下午做完了.说来也巧,我给DASCTF十月赛出的逆向题其中一道也是flutter,不过那题我难度降的相当之低啦,不知道有多少人做出来了呢~ 还原函数名 flutter逆向的一大难点就是不知道lib…

lvs集群中NAT模式

群集的含义 由多台主机构成&#xff0c;但对外表现为一个整体&#xff0c;只提供一个访问入口&#xff0c;相当于一台大型的计算机。 横向发展:放更多的服务器&#xff0c;有调度分配的问题。 垂直发展&#xff1a;升级单机的硬件设备&#xff0c;提高单个服务器自身功能。 …

论文阅读:Scalable Diffusion Models with Transformers

Scalable Diffusion Models with Transformers 论文链接 介绍 传统的扩散模型基于一个U-Net骨架&#xff0c;这篇文章提出了一种新的扩散模型结构&#xff0c;将U-Net替换为一个transformer&#xff0c;并将这种结构称为Diffusion Transformers (DiTs)。他们还发现&#xff…

Codesys.运动控制电子齿轮

文章目录 一. 电子齿轮概念应用 二. 电子齿轮耦合功能块 2.1. MC_GearIn 2.2. MC_GearInPos 2.3. MC_GearOut 三. 电子齿轮案例 3.1. 样例介绍 3.2. 引入虚轴 3.3. 程序框架 3.4. 程序编写 3.5. 程序监控 一. 电子齿轮概念应用 在很多应用场景中有多个牵引轴每个牵引…

.NET开源的两款第三方登录整合库

前言 我相信做开发的同学应该都对接过各种各样的第三方平台的登录授权&#xff0c;来获取用户信息&#xff08;如&#xff1a;微信登录、支付宝登录、QQ登录、GitHub登录等等&#xff09;。今天大姚分享两款.NET开源的第三方登录整合库。 MrHuo.OAuth MrHuo.OAuth是.NET项目…

Tcp标志位 笔记240309

Tcp标志位 TCP&#xff08;传输控制协议&#xff09;的标志位是用于指示TCP报文段中特定控制信息的位字段。这些标志位存在于TCP报头中&#xff0c;用于控制TCP连接的建立、数据传输和终止等过程。以下是TCP标志位的详细说明&#xff1a; SYN&#xff08;同步标志&#xff09;…

Android应用界面

概述&#xff1a;由于学校原因&#xff0c;估计会考&#xff0c;曹某人就浅学一下。 目录 View概念 创建和使用布局文件 相对布局 线性布局 水平线性布局 垂直线性布局 表格布局 帧布局 扁平化布局 Android控件详解 AdapterView及其子类 View概念 安卓中的View是所…

设计模式-行为型设计模式-命令模式

命令模式&#xff08;Command&#xff09;&#xff0c;将一个请求封装为一个对象&#xff0c;从而使你可用不同的请求对客户进行参数化&#xff1b;对请求排队或记录请求日志&#xff0c;以及支持可撤销的操作。[DP] // 命令接口 interface Command {void execute(); }// 具体命…

隐私与创新的交汇点:Partisia Blockchain 重绘技术蓝图

正当我们在这个信息泛滥的时代寻找稳固的信任锚点时&#xff0c;区块链技术应运而生&#xff0c;然而&#xff0c;正如任何科技革命都会遇到的挑战&#xff0c;一个重要的问题摆在了我们面前&#xff1a;如何在不牺牲个人隐私的前提下&#xff0c;享受区块链技术带来的好处&…

【漏洞复现】帮管客 CRM jiliyu SQL注入漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…

MySQL安装使用(mac)

目录 一、下载MySQL 二、环境变量 三、启动 MySql 四、初始化密码设置 一、下载MySQL 打开 MySql 官方下载页面 我是macOS12&#xff0c;所以选择了8.0.30 下载完成之后&#xff0c;打开安装&#xff0c;一直下一步安装完成&#xff0c;在最后安装完成时&#xff0c;会弹出…

基于Docker搭建Maven私服仓库(Linux)详细教程

文章目录 1. 下载镜像并启动容器2. 配置Nexus3. 配置本地Maven仓库 1. 下载镜像并启动容器 下载Nexus3镜像 docker pull sonatype/nexus3查看Nexus3镜像是否下载成功 docker images创建Nexus3的挂载文件夹 mkdir /usr/local/nexus-data && chown -R 200 /usr/local…

基于SSM的校园疫情管理系统的设计与实现(有报告)。Javaee项目。ssm项目。

演示视频&#xff1a; 基于SSM的校园疫情管理系统的设计与实现&#xff08;有报告&#xff09;。Javaee项目。ssm项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;…

鸿蒙开发-UI-动画-页面内动画

鸿蒙开发-UI-组件2 鸿蒙开发-UI-组件3 鸿蒙开发-UI-气泡/菜单 鸿蒙开发-UI-页面路由 鸿蒙开发-UI-组件导航-Navigation 鸿蒙开发-UI-组件导航-Tabs 鸿蒙开发-UI-图形-图片 鸿蒙开发-UI-图形-绘制几何图形 鸿蒙开发-UI-图形-绘制自定义图形 文章目录 前言 一、概述 二、页面内…

【LakeHouse】LakeHouse 架构指南

LakeHouse 架构指南 1.什么是数据湖&#xff0c;为什么需要数据湖2.数据湖、数据仓库和 LakeHouse 之间有什么区别3.数据湖的组件3.1 存储层 / 对象存储&#xff08;AWS S3、Azure Blob Storage、Google Cloud Storage&#xff09;3.2 数据湖文件格式&#xff08;Apache Parque…

C#,老鼠迷宫问题的回溯法求解(Rat in a Maze)算法与源代码

1 老鼠迷宫问题 迷宫中的老鼠&#xff0c;作为另一个可以使用回溯解决的示例问题。 迷宫以块的NN二进制矩阵给出&#xff0c;其中源块是最左上方的块&#xff0c;即迷宫[0][0]&#xff0c;目标块是最右下方的块&#xff0c;即迷宫[N-1][N-1]。老鼠从源头开始&#xff0c;必须…

R语言 | 复数 相关函数

问题 大家好&#xff0c;我有一个问题&#xff0c;我看到一个函数如下&#xff1a; L2_distance <- function(A, B){rowA <- apply(A*A, 1, sum)matrixA <- matrix(rep(rowA, eachlength(rowA)), nrowlength(rowA), byrowT)rowB <- apply(B*B, 1, sum)matrixB &l…

Elasticsearch:什么是 DevOps?

DevOps 定义 DevOps 是一种现代软件开发方法&#xff0c;它将公司软件开发 (Dev) 和 IT 运营 (Ops) 团队的工作结合起来并实现自动化。 DevOps 提倡这样一种理念&#xff1a;这些传统上独立的团队在协作方面比在孤岛中更有效。 理想情况下&#xff0c;DevOps 团队共同努力改进…