Spring Data JPA关于懒加载的那些事

news2025/1/11 5:12:17

背景

环境

相关环境配置:

  • SpringBoot+PostGreSQL

  • Spring Data JPA

懒加载现象

首先声明一下 application.yml 文件中关于 JPA 的配置:

spring:
  jpa:
    show-sql: true
    hibernate:
      ddl-auto: none
    open-in-view: false
    properties:
      hibernate:
        order_by:
          default_null_ordering: last
        order_inserts: true
        order_updates: true
        generate_statistics: false
        jdbc:
          batch_size: 5000

因某些原因,无法直接贴出相关代码,这里就贴一下自己构建的类似代码:

@Entity
@EqualsAndHashCode(callSuper = true, onlyExplicitlyIncluded = true)
@Setter
@Getter
@AllArgsConstructor
@NoArgsConstructor
@SuperBuilder
public class User extends BaseDomain {

  private String name;

  private Integer age;

  private String address;

  @OneToMany(cascade = CascadeType.ALL)
  @JoinColumn(name = "user_id")
  private List<Job> jobs = new ArrayList<>();
}

@Entity
@EqualsAndHashCode(callSuper = true, onlyExplicitlyIncluded = true)
@Setter
@Getter
@AllArgsConstructor
@NoArgsConstructor
@SuperBuilder
public class Job extends BaseDomain {

  private String name;

  @ManyToOne
  @JoinColumn
  private User user;

  private String address;
}

自定义查询语句如下:

@Repository
public interface UserRepository extends JpaRepository<User, Long> {

  @EntityGraph(
      attributePaths = {"jobs"}
  )
  List<User> findByAddress(String address);
}

当我们在 Service 服务中调用类似 findByAddress()的查询方法,断点调试发现查询结果有点意思,明明数据已经能拉出来了,但是在 Hibernate Interceptor 中,并未复制给对应的属性。其次,可以看到得到的对象并不是真正的实体对象,而是实体的代理。

我们来对比一个正常的调用:

很明显,这里返回了想要的结果,而异常的则没有,而且这里得到的是真正的实体对象而非代理。异常的多了一个 $$_hibernate_interceptor 属性,该属性内嵌的属性包含了所需要的数据。

本人所遇到的这个现象并不影响程序的执行,不过成功吸引到我去补充 JPA 中懒加载的知识。

懒加载

代码结构如下:

从代码可以看出,User 和 Job 是一对多的关系,而且我们在 User 类中使用了 @OneToMany 注解,该注解中有个 fetch 属性默认为 FetchType.LAZY,即懒加载。

懒加载(lazy)又叫做延时加载,就是当在真正需要数据的时候,才真正执行数据加载操作。至于为什么要用懒加载呢,就是当我们要访问的数据量过大时,明显用缓存不太合适,因为内存容量有限 ,为了减少并发量,减少系统资源的消耗,我们让数据在需要的时候才进行加载,这时我们就用到了懒加载。

我们以下面代码为例,查看 SQL 执行过程:

  @Transactional
  public List<UserResponse> findByAddress(String address) {
    List<User> users = userRepository.findByAddress(address);
    System.out.println("-----get-----");
    User user = users.get(0);
    List<Job> jobs = user.getJobs();
    return users.stream().map(this::toUserResponse).collect(Collectors.toList());
  }

如果是 Lazy,SQL 日志如下:

如果是 Eager,SQL 日志如下:

从上述输出的 SQL 日志可以看出,不管是 lazy 还是 eager,在读取数据的时候,都会有 N+1问题。

Spring Data JPA 为了简单地提高查询率,引入了 EntityGraph 的概念,可以解决 N+1条SQL的问题。

修改一下 UserRepository

  @EntityGraph(
      attributePaths = {"jobs"}
  )
  List<User> findByAddress(String address);

SQL 输出记录如下所示:

Open Session In View

Open Session In View 简称 OSIV,是为了解决在 mvc 的 controller 中使用了 hibernate 的 lazy load 的属性时 no session 抛出的LazyInitializationException 异常。

关于 LazyInitializationException 异常有四种解决方案,在下文会详细介绍。需要注意的是,通过 OSIV 技术来解决 LazyInitializationException 问题会导致 open 的 session 生命周期过长,它贯穿整个 request,在 view 渲染完之后才能关闭 session 释放数据库连接。另外 OSIV 将 service 层的技术细节暴露到了 controller 层,造成了一定的耦合,因而不建议开启,对应的解决方案就是在 controller 层中使用 response,而非 detached 状态的 entity,所需的数据不再依赖延时加载,在组装 response 的时候根据需要显式查询。

在 SpringBoot 中,配置文件中有这样一个配置:spring.jpa.open-in-view=true,推荐设置为 false。

问题记录

一、Unable to evaluate the expression Method threw 'org.hibernate.LazyInitializationException' exception.

出现场景:使用了懒加载,但没有使用 @EntityGraph,且在配置文件中将 open-in-view 设置为 false

具体原因为:因为当前不存在会话。Hibernate 打开一个会话并关闭它,但是对于“lazy = true”或“fetch = FetchType.LAZY”,这些字段由代理填充。当您尝试查找此类字段的值时,它将尝试使用活动会话访问数据库以检索数据。如果找不到此类会话,则会出现此异常。

解决方案:

1、@OneToMany 中设置 fetch 为 FetchType.EAGER;

2、增加 @EntityGraph,避免 N+1查询问题;

3、在方法上增加 @Transactional,这样可以保持 seesion,使其不被关掉;

4、将 open-in-view 设置为 true,让 session 不被关掉。

二、Cannot call sendError() after the response has been committed

出现场景:

public class User extends BaseDomain {

  private String name;

  private Integer age;

  private String address;

  @OneToMany(cascade = CascadeType.ALL)
  @JoinColumn(name = "user_id")
  private List<Job> jobs = new ArrayList<>();
}

public class Job extends BaseDomain {

  private String name;

  @ManyToOne
  @JoinColumn
  private User user;

  private String address;
}

//UserRepository
@EntityGraph(
  attributePaths = {"jobs"}
)
List<User> findByAddress(String address);

//UserServiceImpl
public List<User> findByAddress2(String address) {
  List<User> users = userRepository.findByAddress(address);
  return users;
}

原因:这是双向关系的一个问题,因为 User 和 Job 类互相引用,在反序列化时,双向引用导致无限递归

解决方法:在@ManyToOne 上方使用@JsonIgnore,@JsonIgnore:直接直接忽略某个属性,以断开无限递归,序列化或反序列化均忽略

扩展学习

@OneToMany

这里着重讲解级联操作。

一方在 oneToMany 上设置的级联保存和更新很好理解,多方会随着一方进行保存和更新。但是级联删除其实只是指一方删除时会把关联的多方数据全部删除,并不能删除一方维护的多方list中 remove 掉的数据。

需要额外注意该注解的这个属性:

orphanRemoval=true

该属性设置为 true,则从关联的多方数据中删除指定数据,即可实现删除操作。

User user=userRepository.findById(1L).get();
ContactInfo deletedContact = user.getContactInfos().get(0);
user.getContactInfos().remove(deletedContact);

JPA 和 Spring Data JPA 有什么区别?

Java Persistence API,有时也称为 JPA,是一个 Java 框架,用于管理使用 Java 平台标准版 (JavaSE) 和 Java 平台企业版 (JavaEE) 的应用程序中的关系数据。

JPA 是标准化 Java 对象映射到关系数据库系统的方式的规范。作为一个规范,JPA 由一组接口(如EntityManagerFactoryEntityManager和注释)组成,可帮助您将 Java 实体对象映射到数据库表。

您可以使用多个 JPA 提供程序,例如 Hibernate、EclipseLink 或 Open JPA。

在这方面的持久性包括三个方面:

  • API 本身,在javax.persistence包中定义。
  • Java 持久性查询语言 (JPQL)。
  • 对象关系元数据。

首先 Spring Data JPA 是 Spring Data 项目的一部分,它可以更轻松地实现基于 JPA 的存储库。

Spring Data JPA 可以与 Hibernate、Eclipse Link 或任何其他 JPA 提供程序一起使用。使用 Spring 或 Java EE 的一个非常有趣的好处是您可以使用@Transactional注解以声明方式控制事务边界。

特征:

  • 支持基于 Spring 和 JPA 构建存储库
  • 支持 QueryDSL 谓词,因此支持类型安全的 JPA 查询
  • 域类的透明审计
  • 分页支持、动态查询执行、集成自定义数据访问代码的能力
  • @Query在引导时验证带注释的查询
  • 支持基于 XML 的实体映射
  • 通过引入基于 JavaConfig 的存储库配置@EnableJpaRepositories

Java Persistence API 中 FetchType LAZY 和 EAGER 的区别?

简单来说就是:

LAZY = fetch when needed
EAGER = fetch immediately

如何将 Hibernate 代理转换为真实的实体对象

如下图所示,从数据集合中筛选出的 metadataFile 不是真正的实体对象,虽然在当前的代码中不影响使用,但是在某些场景下,我们不希望只获取到代理对象。

解决方案:

从 Hibernate ORM 5.2.10开始,您可以这样做:

Object unproxiedEntity = Hibernate.unproxy(proxy);
复制代码

在 Hibernate 5.2.10之前。最简单的方法是使用Hibernate 内部实现提供的unproxy方法:PersistenceContext

Object unproxiedEntity = ((SessionImplementor) session)
                         .getPersistenceContext()
                         .unproxy(proxy);

除此之外,还有一种方法,相较于上面稍微复杂了点:

public static <T> T initializeAndUnproxy(T entity) {
    if (entity == null) {
        throw new 
           NullPointerException("Entity passed for initialization is null");
    }

    Hibernate.initialize(entity);
    if (entity instanceof HibernateProxy) {
        entity = (T) ((HibernateProxy) entity).getHibernateLazyInitializer()
                .getImplementation();
    }
    return entity;
}

最终的效果如下图所示:

Spring Data JPA 中的 CrudRepository 和 JpaRepository 接口有什么区别?

JpaRepository扩展PagingAndSortingRepository而扩展CrudRepository

它们的主要功能是:

  • CrudRepository主要提供CRUD功能。
  • PagingAndSortingRepository提供对记录进行分页和排序的方法。
  • JpaRepository提供一些 JPA 相关的方法,例如刷新持久化上下文和批量删除记录。

由于上面提到的继承, JpaRepository将具有CrudRepositoryPagingAndSortingRepository的所有功能。因此,如果您不需要存储库具有 and 提供的功能JpaRepository,请PagingAndSortingRepository使用CrudRepository.

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

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

相关文章

「6」线性代数(期末复习)

&#x1f680;&#x1f680;&#x1f680;大家觉不错的话&#xff0c;就恳求大家点点关注&#xff0c;点点小爱心&#xff0c;指点指点&#x1f680;&#x1f680;&#x1f680; 目录 第五章 相似矩阵及二次型 &2&#xff09;方阵的特征值与特征向量 &3&#xff…

从transformers开始,哪些工作可以被看成是自然语言处理里程碑式的突破。

文本生成的含义是在某一语言数据基础上对语言的从前到后&#xff08;自监督本身下行目标&#xff09;、两段对齐语言序列&#xff08;相互之间的文本共性矩阵计算&#xff09;分布的研究实现路线。简而言之如何以具有可微可导的向量去寻找攻关语言分布的能力即为当代信息科学与…

Springcloud 集成 Seata1.5.2

一、关于seata1.5.2的安装部署请参考&#xff1a; Seata1.5.2安装配置部署_不知道取啥昵称的博客-CSDN博客 二、springcloud程序集成seata 我这里使用的alibaba-cloud版本为 2.2.6.RELEASE&#xff0c;其对应的seata版本为1.3.0&#xff0c;但是想使用seata1.5.2的版本&…

MongoDB-查找表里面重复的记录

一、背景 项目中使用的是mongodb数据库&#xff0c;在测试数据入库的时候&#xff0c;会根据源数据&#xff0c;然后生成一个自增的id到数据库里面&#xff0c;然后线上和测试环境针对同一条数据的id是不一致的。某些数据又只有id与线上匹配上的时候&#xff0c;才能关联上更多…

CentOS 7转化系统为阿里龙蜥Anolis OS 7

转载&#xff1a;原社区CentOS 7迁移Anolis OS 7迁移手册 一、注意事项 Anolis OS 7生态上和依赖管理上保持跟CentOS7.x兼容&#xff0c;一键式迁移脚本centos2anolis.py&#xff0c;实现CentOS7.x到Anolis OS 7的平滑迁移。 使用迁移脚本前需要注意如下事项&#xff1a; 迁…

Springboot扩展点之CommandLineRunner和ApplicationRunner

Springboot扩展点系列&#xff1a;Springboot扩展点之ApplicationContextInitializerSpringboot扩展点之BeanDefinitionRegistryPostProcessorSpringboot扩展点之BeanFactoryPostProcessorSpringboot扩展点之BeanPostProcessorSpringboot扩展点之InstantiationAwareBeanPostPro…

获取保存在本地的帐户密码,支持浏览器|数据库|邮件|无线|管理员等等

获取保存在本地的帐户密码&#xff0c;支持浏览器|数据库|邮件|无线|管理员等等。 #################### 免责声明&#xff1a;工具本身并无好坏&#xff0c;希望大家以遵守《网络安全法》相关法律为前提来使用该工具&#xff0c;支持研究学习&#xff0c;切勿用于非法犯罪活动…

使用 Nodejs、Express、Postgres、Docker 在 JavaScript 中构建一个 CRUD Rest API

让我们在 JavaScript 中创建一个 CRUD rest API&#xff0c;使用&#xff1a;节点.js表达续集Postgres码头工人码头工人组成介绍这是我们将要创建的应用程序架构的架构&#xff1a;我们将为基本的 CRUD 操作创建 5 个端点&#xff1a;创造阅读全部读一个更新删除我们将使用以下…

python自学之《21天学通Python》(11)——网络编程

第14章 网络编程 网络编程是现代编程主题中的一个重要组成部分&#xff0c;而Python在标准库中就已经提供了丰富的网络编程模块&#xff0c;以支持用户进行编写具有各种网络功能的程序或软件。在Python标准库中&#xff0c;支持底层网络编程的是socket模块&#xff1b;针对特定…

ilasm 和 ildasm编译和反编译工具介绍使用教程

目录前言一、使用 ildasm 反编译 dll 文件二、使用 ilasm 将il文件编译成 dll 或 exe 文件前言 文本讲述怎么通过 ildasm 工具将 dll 文件进行反编译为 il 文件&#xff0c;修改 il 文件后再如何通过 ilasm 工具将 il 文件反编译成 dll 或 exe 文件。 ildasm工具&#xff1a;…

【2023最火教程】5分钟学会接口自动化测试框架

今天&#xff0c;我们来聊聊接口自动化测试。 接口自动化测试是什么&#xff1f;如何开始&#xff1f;接口自动化测试框架如何搭建&#xff1f; 自动化测试 自动化测试&#xff0c;这几年行业内的热词&#xff0c;也是测试人员进阶的必备技能&#xff0c;更是软件测试未来发展…

【蓝桥杯集训5】递推专题(3 / 3)

目录 3777. 砖块 - 递推 1208. 翻硬币 - 递推 95. 费解的开关 - 递推 位运算 3777. 砖块 - 递推 3777. 砖块 - AcWing题库 题目&#xff1a; 思路&#xff1a; 要使所有砖块颜色一致&#xff0c;则要不全B&#xff0c;要不全W 则分情况讨论&#xff1a;全为白色和全为黑色…

MariaDB 成功敲钟上市 | 它与 Navciat 缘起 10 年前

MariaDB 敲钟上市2022 年底&#xff0c;云数据库公司 MariaDB 与 Angel Pond Holdings 公司完成合并&#xff0c;并在纽交所上市。新公司更名为 MariaDB&#xff0c;MySQL 之父奋斗了13年终敲钟。这标志着 MariaDB 开启新篇章。无论从开源还是商业之路&#xff0c;都将成为业内…

整理了十个Python自动化操作,拿走就用

01OS模块相关一、遍历文件夹 批量操作的前提就是对文件夹进行遍历&#xff0c;使用os模块可以轻松的遍历文件夹&#xff0c;os.walk 遍历后产生三个参数&#xff1a; 当前文件夹路径 包含文件夹名称[列表形式] 包含文件名称[列表形式] 代码如下&#xff0c;大家可以根据自己的…

SpringCloudAlibab-nacos

一、介绍注册中心配置中心的整合SpringCloudAlibaba中文地址&#xff1a;https://github.com/alibaba/spring-cloud-alibaba/blob/2.2.x/README-zh.md下载地址&#xff1a;https://github.com/alibaba/nacos/访问&#xff1a;http://localhost:8848/nacos/二、使用1、添加依赖&…

[大邻域算法](MD)VRPTW常见求解算法--代码解析

相关链接&#xff1a; 【路径分割】序列分隔和路径提取的案例【算法】LNS(大邻域算法)和ALNS(自适应大邻域算法)(持更)Python实现(MD)VRPTW常见求解算法——自适应大邻域搜索算法&#xff08;ALNS&#xff09;,本文也是该篇的解析干货 | 自适应大邻域搜索(Adaptive Large Neig…

【每日一题Day121】LC1139最大的以 1 为边界的正方形 | 前缀和数组 + 枚举

最大的以 1 为边界的正方形【LC1139】 给你一个由若干 0 和 1 组成的二维网格 grid&#xff0c;请你找出边界全部由 1 组成的最大 正方形 子网格&#xff0c;并返回该子网格中的元素数量。如果不存在&#xff0c;则返回 0。 写了50分钟写出来了 思路是对的 但就是不够清晰 并且…

ACPI on ARMv8 Servers

文章目录前言一、Why ACPI on ARM&#xff1f;二、Kernel Compatibility三、Relationship with Device Tree四、Booting using ACPI tables五、ACPI Detection六、Device Enumeration七、Driver Recommendations参考资料前言 ARM64处理器除了可以用设备树&#xff08;DT&#…

V90伺服驱动器设置IP地址和PN设备名称的具体方法(2种)

V90伺服驱动器设置IP地址和PN设备名称的具体方法(2种) 1. 通过V-ASSISTANT软件进行配置 首先下载并安装V-ASSISTANT软件,然后将V90通过网线连接到电脑上,注意此时电脑使用的网卡,不能选择无线网卡, SINAMICS-V90伺服调试软件V-ASSISTANT_V1.07.01 打开V-ASSISTANT软件,…

Java 基础面试题——面向对象

目录1.面向对象和面向过程有什么区别&#xff1f;2.面向对象的有哪些特征?3.静态变量和实例变量有什么区别&#xff1f;4.Java 对象实例化顺序是怎样的&#xff1f;5.浅拷贝和深拷贝的区别是什么&#xff1f;5.1.浅拷贝5.2.深拷贝5.3.总结6.Java 中创建对象的方式有哪几种&…