Spring Data JPA的一对一、LazyInitializationException异常、一对多、多对多操作

 Spring Data JPA系列

1、SpringBoot集成JPA及基本使用

2、Spring Data JPA Criteria查询、部分字段查询

3、Spring Data JPA数据批量插入、批量更新真的用对了吗

4、Spring Data JPA的一对一、LazyInitializationException异常、一对多、多对多操作

前言

通过前三篇Spring Data JPA的博文,相信大家对JPA有了一定的了解,然而前面的文章都只介绍单个表,这一篇,将给大家分享一下多表关系的定义及相关操作。在开始介绍之前,先了解一些理论知识。

CascadeType

JPA框架中的cascade属性用于指定实体之间的级联操作。级联操作是指当一个实体的状态发生改变时,关联的其他实体是否同时发生改变。简单理解:cascade用于设置当前实体是否能够操作关联的另一个实体的权限。

cascade的取值为CascadeType枚举类型

1)CascadeType.PERSIST:持久化操作时会级联执行(cascade是用于设置实体的权限,所以对应的持久化操作也是针对实体对象的持久化,对应EntityManager的persist()方法操作,而如果是通过JPQL、HQL或Sql实现的持久化,则不会级联。下同。);

2)CascadeType.MERGE:合并(更新)操作时级联执行;

3)CascadeType.REMOVE:删除操作时级联执行;

4)CascadeType.REFRESH:刷新操作时级联执行;

5)CascadeType.DETACH:级联脱管/游离操作,如果要删除一个实体,但是它有外键无法删除,需要这个级联权限。它会撤销所有相关的外键关联;

6)CascadeType.ALL:上面的5种操作都会级联执行;

当通过注解来映射持久化类时,如果希望使用底层Hibernate的一些级联特性,还可以使用CascadeType类的一些常量,例如:

1)org.hibernate.annotations.CascadeType.LOCK:当通过底层Session的lock()方法把当前游离对象加入到持久化缓存中时,会把所有关联的游离对象也加入到持久化缓存中;

2)org.hibernate.annotations.CascadeType.REPLICATE:当通过底层Session的replicate()方法复制当前对象时,会级联复制所有关联的对象;

3)org.hibernate.annotations.CascadeType.SAVE_UPDATE:当通过底层Session的save()、update()及saveOrUpdate()方法来保存或更新当前对象时,会级联保存所有关联的新建的临时对象,并且级联更新所有关联的游离对象;

注:在实际开发中谨慎使用cascade属性,以免对数据库造成不可预知的影响。

一对一

一对一关系,一个表中的记录与另一个表中的记录之间存在唯一的对应关系。以商品为例,一件商品只有一个详情信息。

2.1 实体类

package com.jingai.jpa.dao.entity;import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import lombok.Data;import javax.persistence.*;
import java.util.Date;@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_goods")
public class GoodsEntity {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String name;private String subtitle;private Long classificationId;private Date createTime;@Transientprivate String createTimeStr;@OneToOne(cascade = {CascadeType.MERGE, CascadeType.REMOVE})@JoinColumn(name = "id", referencedColumnName = "id")private GoodsDetailEntity detail;}

@Transient注解:该注解用于标注对应的属性不在数据库表中;

@OneToOne注解:用于实体上的注解,表示一对一关系。此处商品详情的数据变更同商品并存,此处的cascade可以设置CascadeType.MERGE和CascadeType.REMOVE;

@JoinColumn注解:标注实体类与数据库的对应关系。主要可选属性如下:

1)name:定义被标注属性在数据库表中所对应的字段的名称;

2)unique:定义被标注属性在数据库表中的值是否唯一,默认为false;

3)insertable:表示在使用“Insert”脚本插入数据时,是否需要插入被标注属性的值,默认为true;

4)updatable:表示在使用“Update”脚本插入数据时,是否需要更新被标注属性的值,默认为true;

5)referencedColumnName:定义所关联表中的字段名;

6)table:定义包含当前字段的表名;

package com.jingai.jpa.dao.entity;import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import lombok.Data;import javax.persistence.*;@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_goods_detail")
public class GoodsDetailEntity {@Idprivate Long id;private String detailDescribe;private String pictures;}

2.2 Repository类

package com.jingai.jpa.dao;import com.jingai.jpa.dao.entity.GoodsEntity;
import org.springframework.data.jpa.repository.support.JpaRepositoryImplementation;public interface GoodsRepository extends JpaRepositoryImplementation<GoodsEntity, Long> {
}
package com.jingai.jpa.dao;import com.jingai.jpa.dao.entity.GoodsDetailEntity;
import org.springframework.data.jpa.repository.support.JpaRepositoryImplementation;public interface GoodsDetailRepository extends JpaRepositoryImplementation<GoodsDetailEntity, Long> {
}

2.3 Service类

package com.jingai.jpa.service;import com.jingai.jpa.dao.GoodsDetailRepository;
import com.jingai.jpa.dao.GoodsRepository;
import com.jingai.jpa.dao.entity.GoodsDetailEntity;
import com.jingai.jpa.dao.entity.GoodsEntity;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;import javax.annotation.Resource;
import java.util.Date;
import java.util.Optional;@Service
public class GoodsService {@Resourceprivate GoodsRepository goodsRepository;@Resourceprivate GoodsDetailRepository goodsDetailRepository;@Transactionalpublic GoodsEntity save(GoodsEntity entity, GoodsDetailEntity detail) {entity.setCreateTime(new Date());entity = goodsRepository.save(entity);detail.setId(entity.getId());detail = goodsDetailRepository.save(detail);entity.setDetail(detail);return entity;}public GoodsEntity get(long id) {// getById()使用懒加载的方式访问数据库,只有在真正访问GoodsEntity的才会真正执行数据库访问return goodsRepository.getById(id);}public GoodsEntity find(long id) {// findById()是立即访问数据库查询数据Optional<GoodsEntity> entity = goodsRepository.findById(id);return entity.isPresent() ? entity.get() : null;}/*** 修改。由于在GoodsEntity中的GoodsDetailEntity的@OneToOne注解配置了CascadeType.MERGE,修改更新会级联执行*/@Transactionalpublic GoodsEntity update(GoodsEntity goods, GoodsDetailEntity detail) {detail.setId(goods.getId());goods.setDetail(detail);goods.setCreateTime(new Date());goods = goodsRepository.save(goods);return goods;}}

说明:此处为了讲解方便,不先定义接口后定义实现类。

2.4 Controller类

package com.jingai.jpa.controller;import com.jingai.jpa.dao.entity.GoodsDetailEntity;
import com.jingai.jpa.dao.entity.GoodsEntity;
import com.jingai.jpa.service.GoodsService;
import com.jingai.jpa.util.ResponseUtil;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.annotation.Resource;
import java.text.SimpleDateFormat;
import java.util.Map;@RestController
@RequestMapping("goods")
public class GoodsController {@Resourceprivate GoodsService goodsService;@PostMapping("save")public Map<String, Object> save(GoodsEntity goods, GoodsDetailEntity detail) {goods = goodsService.save(goods, detail);goods.setCreateTimeStr(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(goods.getCreateTime()));return ResponseUtil.success(goods);}@GetMapping("get")public Map<String, Object> get(long id) {GoodsEntity entity = goodsService.get(id);return ResponseUtil.success(entity);}@GetMapping("find")public Map<String, Object> find(long id) {GoodsEntity entity = goodsService.find(id);return ResponseUtil.success(entity);}@PostMapping("update")public Map<String, Object> update(GoodsEntity goods, GoodsDetailEntity detail) {goods = goodsService.update(goods, detail);goods.setCreateTimeStr(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(goods.getCreateTime()));return ResponseUtil.success(goods);}}

2.5 访问接口,结果如下:

2.6 LazyInitializationException异常

当访问

http://localhost:8080/goods/get?id=14

时,系统会报Method threw ‘org.hibernate.LazyInitializationException‘ exception. Cannot evaluate异常,原因是JpaRepository.getById()方法是懒加载,访问该方法时,返回一个对应实体的引用,而该引用是没有值的,此时创建了一个临时的session,并没有真正访问数据库,并立即关闭了session。在Controller层中访问该引用的信息时才真正执行数据库的访问,此时session已关闭,所以报了上面的异常,提示no session。

解决方法:

方法一:在application.yml中添加spring.jpa.open-in-view=true。这个配置在SpringBoot集成JPA及基本使用-CSDN博客中有讲解,建议关闭。而且该配置只能解决通过controller层访问引起的懒加载问题;

方法二:在application.yml中添加spring.jpa.properties.hibernate.enable_lazy_load_no_trans=true。该方法既能解决通过Controller层访问引起的懒加载,也能解决定时任务等访问引起的懒加载问题;

当然也可以使用CrudRepository.findById()这个接口,立即访问数据库。

一对多

一对多关系,一个表中的一条记录与另一个表中的多条记录之间相对应。如一个会员有多个地址。

3.1 实体类

package com.jingai.jpa.dao.entity;import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import lombok.Data;import javax.persistence.*;
import java.util.Date;@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_member")
public class MemberEntity {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private int id;private String name;private String sex;private Date createTime;
}
package com.jingai.jpa.dao.entity;import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import lombok.Data;import javax.persistence.*;
import java.util.Date;@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_address")
public class AddressEntity {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private int id;@Column(name = "member_id")private int memberId;@ManyToOne@JoinColumn(name = "member_id", referencedColumnName = "id", updatable = false, insertable = false)private MemberEntity member;private String province;private String city;private String address;private String phone;private Date createTime;
}

@ManyToOne注解:地址与会员是多对一的关系,所以此处添加了@ManyToOne注解,表示多对一,不添加cascade配置,因为会员也可以没有地址,它们之前没有必然关联;

@JoinColumn注解:此处的updateable为false,insertable也为false。即对Address表的修改不会影响到Member表;

3.2 Repository类

package com.jingai.jpa.dao;import com.jingai.jpa.dao.entity.MemberEntity;
import org.springframework.data.jpa.repository.support.JpaRepositoryImplementation;public interface MemberRepository extends JpaRepositoryImplementation<MemberEntity, Long> {
}
package com.jingai.jpa.dao;import com.jingai.jpa.dao.entity.AddressEntity;
import org.springframework.data.jpa.repository.support.JpaRepositoryImplementation;public interface AddressRepository extends JpaRepositoryImplementation<AddressEntity, Long> {
}

3.3 Service类

package com.jingai.jpa.service;import com.jingai.jpa.common.form.AddressForm;
import com.jingai.jpa.dao.AddressRepository;
import com.jingai.jpa.dao.entity.*;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.data.jpa.domain.Specification;
import org.springframework.stereotype.Service;
import org.springframework.util.StringUtils;import javax.annotation.Resource;
import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.criteria.CriteriaQuery;
import javax.persistence.criteria.Predicate;
import javax.persistence.criteria.Root;
import java.util.ArrayList;
import java.util.List;@Service
public class AddressService {@Resourceprivate AddressRepository addressRepository;public Page<AddressEntity> listByPage(AddressForm form) {// 创建一个Specification,实现接口中的toPredicate()方法,该方法返回一个PredicateSpecification<AddressEntity> specification = new Specification<AddressEntity>() {@Overridepublic Predicate toPredicate(Root<AddressEntity> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {List<Predicate> predicates = new ArrayList<>(8);// 通过会员名称查询if(StringUtils.hasText(form.getName())) {// 先通过AddressEntity_.member定位到MemberEntity,然后再定位到MemberEntity_.namepredicates.add(criteriaBuilder.like(root.get(AddressEntity_.member).get(MemberEntity_.name), "%" + form.getName() + "%"));}if(StringUtils.hasText(form.getPhone())) {predicates.add(criteriaBuilder.like(root.get(AddressEntity_.PHONE), "%" + form.getPhone() + "%"));}if(form.getStartDate() != null) {predicates.add(criteriaBuilder.greaterThanOrEqualTo(root.get(AddressEntity_.createTime), form.getStartDate()));}if(form.getEndDate() != null) {predicates.add(criteriaBuilder.lessThanOrEqualTo(root.get(AddressEntity_.createTime), form.getEndDate()));}return query.where(predicates.toArray(new Predicate[predicates.size()])).getRestriction();}};// 创建排序字段,可设置多个Sort sort = Sort.by(Sort.Direction.DESC, ProductEntity_.createTime.getName());Pageable pageable = PageRequest.of(form.getPageIndex(), form.getPageSize(), sort);// 使用JpaSpecificationExecutor的findAll()方法,只能返回实体类的集合return addressRepository.findAll(specification, pageable);}}

此处重点讲解AddressService中使用Criteria查询某个会员名称的地址信息。对Criteria查询不清楚的,可以看

Spring Data JPA Criteria查询、部分字段查询-CSDN博客

示例中的AddressForm在该篇博文也有讲解到。

此处可以通过JPA的元模式,很方便的实现表的关联查询。

3.4 Controller类

@RestController
@RequestMapping("address")
public class AddressController {@Resourceprivate AddressService addressService;@GetMapping("search")public Map<String, Object> search(AddressForm form) {return ResponseUtil.success(addressService.listByPage(form));}}

访问接口后显示效果如下:

多对多

多对多关系,指两个表中的记录可以相互对应。如一个学生可以选择多门课程,一门课程可以被多个学生选择。针对多对多关系的场景,通常使用中间表进行关联。

@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_student")
public class Student {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String name;@ManyToMany@JoinTable(name = "student_course", joinColumns = {@JoinColumn(name = "student_id")}, inverseJoinColumns = {@JoinColumn(name = "course_id")})private List<Course> courses;}
@Data
@Entity
@JsonIgnoreProperties(value = { "hibernateLazyInitializer"})
@Table(name = "tb_student")
public class Course {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String name;@ManyToMany(mappedBy = "courses")private List<Student> students;}

@ManyToMany注解:用于标注多对多关系。mappedBy为被注解的多的实体类的属性字段;

@JoinTable注解:name为关联表的名称;joinColumns:关联student表的id;inverseJoinColumns:关联Course表的id;

结尾

限于篇幅,Spring Data JPA的一对一、一对多、多对多操作就分享到这里。

关于本篇内容你有什么自己的想法或独到见解,欢迎在评论区一起交流探讨下吧

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

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

相关文章

GNU Radio创建FFT、IFFT C++ OOT块

文章目录 前言一、GNU Radio官方FFT弊端二、创建自定义的 C OOT 块1、创建 OOT 模块2、创建 OOT 块3、修改 C 和 CMAKE 文件4、编译及安装 OOT 块 三、测试1、grc 图2、运行结果①、时域波形对比②、频谱图对比 四、资源自取 前言 GNU Radio 自带的 FFT 模块使用起来不是很方便…

RT-DETR-20240507周更说明|更新Inner-IoU、Focal-IoU、Focaler-IoU等数十种IoU计算方式

RT-DETR改进专栏|包含主干、模块、注意力、损失函数等改进 专栏介绍 本专栏包含模块、卷积、检测头、损失等深度学习前沿改进,目前已有改进点70&#xff01;每周更新。 20240507更新说明&#xff1a; ⭐⭐ 更新CIoU、DIoU、MDPIoU、GIoU、EIoU、SIoU、ShapeIou、PowerfulIoU、…

04-28 周日 FastAPI Post请求同时传递文件和普通参数

04-28 周日 FastAPI Post请求同时传递文件和普通参数 时间版本修改人描述04-28 周日V0.1宋全恒新建文档2024年5月6日14:20:05V1.0宋全恒完成文档的传递 简介 由于在重构FastBuild的时候&#xff0c;为了支持TLS是否启用&#xff0c;在接口中需要同时传递文件参数和其他参数&am…

【Vue3】Ref与Reactive

3.1【ref 创建&#xff1a;基本类型的响应式数据】 作用&#xff1a;定义响应式变量。语法&#xff1a;let xxx ref(初始值)。返回值&#xff1a;一个RefImpl的实例对象&#xff0c;简称ref对象或ref&#xff0c;ref对象的value属性是响应式的。注意点&#xff1a; JS中操作数…

docker 指定根目录 迁移根目录

docker 指定根目录 迁移根目录 1、问题描述2、问题分析3、解决方法3.1、启动docker程序前就手动指定docker根目录为一个大的分区(支持动态扩容)&#xff0c;事前就根本上解决根目录空间不够问题3.1.0、方法思路3.1.1、docker官网安装文档3.1.2、下载docker安装包3.1.3、安装doc…

网络安全之交换基础

交换属于二层技术。路由器&#xff08;router&#xff09;是三层设备&#xff0c;可以基于IP地址转发&#xff0c;但需要路由表来记录。 交换机&#xff08;switch&#xff09;是二层设备&#xff0c;网桥&#xff08;switch&#xff09;也是二层设备&#xff0c;这两个都是基…

zabbix监控方式(zabbix-trapper)

中文&#xff1a;zabbix采集器&#xff0c;即zabbix sender 。 Zabbix-Trapper 监控方式可以一次批量发送数据给Zabbix Server&#xff0c;与主动模式不同&#xff0c;Zabbix-Trapper 可以让用户控制数据的发送&#xff0c;而不用Zabbix-Agent进程控制&#xff0c;这意味着可以…

【算法小白周赛1A】分析 - 题解与代码

题目链接&#xff1a;https://www.starrycoding.com/problem/155 题目描述 小可可最近在学数学运算&#xff01;他希望考考你&#xff0c;给你两个整数 A , B A,B A,B&#xff0c;询问 A B A\times B AB 是否是偶数。 注意&#xff0c;可能存在前导 0 0 0&#xff0c;比如…

与Apolo共创生态: Apollo X企业自动驾驶解决方案的亮点

文章目录 前言技术革新的里程碑Apollo X企业自动驾驶解决方案的亮点Application X企业预制套件的多场景覆盖Studio X企业协同工具链的全周期支持第一阶段&#xff1a;上机系统构建第二阶段&#xff1a;POC搭建第三阶段&#xff1a;规模运营小结 共创生态&#xff0c;共享未来共…

Sermant在异地多活场景下的实践

Sermant社区在1.3.0和1.4.0版本相继推出了消息队列禁止消费插件和数据库禁写插件&#xff0c;分别用于解决异地多活场景下的故障切流和保护数据一致性问题。本文将对Sermant在异地多活场景下的实践进行剖析。 一、异地多活 1.1 什么是异地多活 对于一个软件系统&#xff0c;…

互联网十万个为什么之什么是云计算

云计算是一种通过互联网提供计算资源和服务的技术。它允许用户随时随地访问和使用云平台上的数据、软件和硬件资源。在数字化时代&#xff0c;互联网已经成为基础设施。云计算使得数据中心能够像一台计算机一样去工作。通过互联网将算力以按需使用、按量付费的形式提供给用户&a…

双热点的王炸组合!损失函数+Attention,精度与速度上实现SOTA!

损失函数注意力机制在深度学习领域是一个热门研究方向&#xff0c;它可以提高模型的性能和泛化能力&#xff0c;帮助我们构建更加精确且高效的模型。 具体来说&#xff1a; 通过结合注意力机制的聚焦能力和损失函数的优化指导&#xff0c;模型能够更精确地捕捉数据中的关键信息…

如何从Windows 10电脑远程登录Ubuntu系统

要从Windows 10电脑远程登录Ubuntu系统&#xff0c;您可以使用以下步骤&#xff1a; 在Ubuntu上安装xRDP: 首先&#xff0c;在Ubuntu电脑上打开终端&#xff0c;然后输入以下命令来安装xRDP服务&#xff1a; sudo apt update sudo apt install xrdpxRDP是一个开源的远程桌面协议…

ES6新特性

1.let声明变量 不允许重复声明、没有变量提升 暂存性死区 let a 1 function test(){console.log(a)//undefinedlet a2 } test() 不与顶层对象挂钩 let a 1 console.log(window.a)//undefined 块级作用域 var olidocument.querySelectorAll("ul li") for(var i0…

使用pandas的merge()和join()函数进行数据处理

目录 一、引言 二、pandas的merge()函数 基本用法 实战案例 三、pandas的join()函数 基本用法 实战案例 四、merge()与join()的比较与选择 使用场景&#xff1a; 灵活性&#xff1a; 选择建议&#xff1a; 五、进阶案例与代码 六、总结 一、引言 在数据分析和处理…

物联网技术、测试要点和测试标准

物联网定义 物联网&#xff1a;利用嵌入式电子设备、微芯片等连接车辆、家电、医疗设备&#xff0c;以收集和交换不同类型的数据&#xff0c;被称为物联网。借助物联网&#xff0c;用户能够远程控制设备&#xff0c;可以实现不同设备的互联。在现实生活中物联网开始有越来越多…

【消息队列】MQ介绍

MQ MQ&#xff08;MessageQueue&#xff09;&#xff0c;中文是消息队列&#xff0c;就是存放消息的队列&#xff0c;也是下面提到的事件驱动架构中的Broker 同步调用的优点&#xff1a; 时效性强&#xff0c;可以立即得到结果 同步调用的问题&#xff1a; 耦合度高性能和吞吐…

2022 年全国职业院校技能大赛高职组云计算赛项试卷(容器云)

#需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包…

自然语言(NLP)

It’s time for us to learn how to analyse natural language documents, using Natural Language Processing (NLP). We’ll be focusing on the Hugging Face ecosystem, especially the Transformers library, and the vast collection of pretrained NLP models. Our proj…