多年的教训:根据DDD设计原则改变JPA/Hibernate的使用方式
liebian365 2024-10-23 13:51 33 浏览 0 评论
我(lorenzo)最近一直在更新一些培训材料,思考JPA更好的教学方法和讨论方式。我一直在思考的一件事是我们通常是如何使用JPA?这里结合我所经历的(和观察到的)痛苦,应该如何改变传统使用方式?
JPA通常被视为一组注释(或XML文件),它们提供O/R(对象关系)映射信息。大多数开发人员认为他们知道和使用的映射注释越多,他们得到的好处就越多。但是在过去的几年里,与中小规模的巨石/单体/整体系统(大约有200张表/实体)的搏斗教会了我一些别的东西。
教训:
- 按ID引用实体(仅映射聚合中的实体关系)
- 不要让JPA窃取你的ID(尽可能避免@GeneratedValue)
- 使用特殊join来join不相关的实体
按标识符ID引用实体
仅映射DDD聚合中的实体关系。
传统 JPA或Hibernate教程(和培训)通常会涵盖所有可能的实体关系映射。在教学基本映射之后,许多映射将从简单的单向@manytone映射开始。然后继续双向@OneToMany和@ManyToOne。不幸的是,大多数情况下,他们没有明确指出,这种映射关系不是很好。因此,初学者在完成训练时往往会认为,不映射相关实体是错误的。他们错误地认为外键字段必须映射为相关实体。
Bash
|
将上面@ManyToOne 应该改为@Column,将相关实体的主键映射为一个字段即可:
Bash
|
映射所有实体关系会增加了不必要的遍历的机会,这通常会导致不必要的内存消耗。这也会导致不必要的EntityManager操作级联。
如果您只处理少数几个实体/表,这可能并不多。但是当与几十个(如果不是几百个)实体一起工作时,它就变成了维护的噩梦。
何时映射相关实体?
仅当相关实体位于聚合中时才映射它们(在DDD中)。
聚合是领域驱动设计中的一种模式。DDD聚合是可以作为单个单元处理的域对象的集群。例如订单及其行项目,它们将是单独的对象,但将订单(及其行项目)视为单个聚合非常有用。
https://martinfowler.com/bliki/DDD_Aggregate.html
|
更现代的聚合设计方法提倡在聚合之间进行更干净的分离。通过存储聚合根的ID(唯一标识符)而不是完整的引用来引用聚合根是一种很好的做法。
如果我们展开上面的简单订单示例,那么行项目(OrderItem类)不应该有到产品的@ManyToOne映射,相反,它应该只有产品的ID:
|
但是…如果产品(聚合根实体)的@Id字段映射为@GeneratedValue呢?我们是否必须先持久化/flush刷新,然后使用生成的ID值?
那么,join呢?我们还能在JPA中Join这些实体吗?
别让JPA偷走你的标识
使用@GeneratedValue最初可能会使映射简单易用。但是,当您开始通过ID(而不是通过映射关系)引用其他实体时,这将成为一个挑战。
如果产品(聚合根实体)的@Id字段映射为@GeneratedValue,则调用getId()可能返回null。当它返回null时,行项目(OrderItem类)将无法引用它!
在所有实体都有一个非空Id字段的环境中,按Id引用任何实体都会变得更容易。此外,始终具有非空的Id字段,使得equals(Object)和hashCode()更容易实现。
因为所有Id字段都显式初始化,所以所有(聚合根)实体都有一个接受Id字段值的公共构造函数。可以添加一个受保护的no-args构造函数来让JPA满意。
|
在写这篇文章的时候,我发现了James Brundege的一篇文章(2006年发布的), Don't Let Hibernate Steal Your Identity (感谢Wayback Machine),他说,不要让Hibernate管理你的Id。但愿我早点听他的劝告。
但要小心!当使用Spring Data JPA save()保存一个在其@Id字段上不使用@GeneratedValue的实体时,在预期的INSERT之前会发出一个不必要的SQL SELECT。这是由于SimpleJpaRepository的save()方法(如下所示)。它依赖于@Id字段(非空值)的存在来确定是调用persist(Object)还是merge(Object)。
|
精明的读者会注意到,如果@Id字段从不为null,save()方法将始终调用merge()。这会导致不必要的SQL SELECT(在预期的INSERT之前)。
幸运的是,解决方法很简单-实现 Persistable<ID>.
|
使用特殊Join连接来join不相关的实体
那么,连接join呢?既然我们通过ID引用了其他实体,那么如何在JPA中连接join不相关的实体呢?
在jpa2.2版本中,不相关的实体不能连接。但是,我无法确认这是否已经成为3.0版的标准,在3.0版中,所有javax.persistence引用都被重命名为jakarta.persistence。
给定OrderItem实体,缺少@manytone映射会导致它无法与产品实体联接。
|
值得庆幸的是,Hibernate 5.1.0+(2016年发布)和EclipseLink 2.4.0+(2012年发布)一直在支持无关实体的连接。这些连接也称为特殊连接 ad-hoc joins。
|
另外,这也是一个API问题(支持两个根实体的JOIN/ON)。我真的希望它能很快成为一种标准。
相关推荐
- go语言也可以做gui,go-fltk让你做出c++级别的桌面应用
-
大家都知道go语言生态并没有什么好的gui开发框架,“能用”的一个手就能数的清,好用的就更是少之又少。今天为大家推荐一个go的gui库go-fltk。它是通过cgo调用了c++的fltk库,性能非常高...
- 旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗
-
这几天老毛桃整理了几个微型Linux发行版,准备分享给大家。要知道可供我们日常使用的Linux发行版有很多,但其中的一些发行版经常会被大家忽视。其实这些微型Linux发行版是一种非常强大的创新:在一台...
- codeblocks和VS2019下的fltk使用中文
-
在fltk中用中文有点问题。英文是这样。中文就成这个样子了。我查了查资料,说用UTF-8编码就行了。edit->Fileencoding->UTF-8然后保存文件。看下下边的编码指示确...
- FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库
-
FLTK(FastLightToolkit)是一个轻量级的跨平台GUI库,特别适用于开发需要快速、高效且简单界面的应用程序。本文将介绍Python中的FLTK库,包括其特性、应用场景以及如何通过代...
- 中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux
-
IT之家1月29日消息,去年6月份,中科院大学教授、中科院计算所研究员包云岗,发布了开源高性能RISC-V处理器核心——香山。近日,包云岗在社交平台晒出图片,香山芯片已流片,回片后...
- Linux 5.13内核有望合并对苹果M1处理器支持的初步代码
-
预计Linux5.13将初步支持苹果SiliconM1处理器,不过完整的支持工作可能还需要几年时间才能完全完成。虽然Linux已经可以在苹果SiliconM1上运行,但这需要通过一系列的补丁才能...
- Ubuntu系统下COM口测试教程(ubuntu port)
-
1、在待测试的板上下载minicom,下载minicom有两种方法:方法一:在Ubuntu软件中心里面搜索下载方法二:按“Ctrl+Alt+T”打开终端,打开终端后输入“sudosu”回车;在下...
- 湖北嵌入式软件工程师培训怎么选,让自己脱颖而出
-
很多年轻人毕业即失业、面试总是不如意、薪酬不满意、在家躺平。“就业难”该如何应对,参加培训是否能改变自己的职业走向,在湖北,有哪些嵌入式软件工程师培训怎么选值得推荐?粤嵌科技在嵌入式培训领域有十几年经...
- 新阁上位机开发---10年工程师的Modbus总结
-
前言我算了一下,今年是我跟Modbus相识的第10年,从最开始的简单应用到协议了解,从协议开发到协议讲解,这个陪伴了10年的协议,它一直没变,变的只是我对它的理解和认识。我一直认为Modbus协议的存...
- 创建你的第一个可运行的嵌入式Linux系统-5
-
@ZHangZMo在MicrochipBuildroot中配置QT5选择Graphic配置文件增加QT5的配置修改根文件系统支持QT5修改output/target/etc/profile配置文件...
- 如何在Linux下给zigbee CC2530实现上位机
-
0、前言网友提问如下:粉丝提问项目框架汇总下这个网友的问题,其实就是实现一个网关程序,内容分为几块:下位机,通过串口与上位机相连;下位机要能够接收上位机下发的命令,并解析这些命令;下位机能够根据这些命...
- Python实现串口助手 - 03串口功能实现
-
串口调试助手是最核心的当然是串口数据收发与显示的功能,pzh-py-com借助的是pySerial库实现串口收发功能,今天痞子衡为大家介绍pySerial是如何在pzh-py-com发挥功能的。一、...
- 为什么选择UART(串口)作为调试接口,而不是I2C、SPI等其他接口
-
UART(通用异步收发传输器)通常被选作调试接口有以下几个原因:简单性:协议简单:UART的协议非常简单,只需设置波特率、数据位、停止位和校验位就可以进行通信。相比之下,I2C和SPI需要处理更多的通...
- 同一个类,不同代码,Qt 串口类QSerialPort 与各种外设通讯处理
-
串口通讯在各种外设通讯中是常见接口,因为各种嵌入式CPU中串口标配,工业控制中如果不够还通过各种串口芯片进行扩展。比如spi接口的W25Q128FV.对于软件而言,因为驱动接口固定,软件也相对好写,因...
- 嵌入式linux为什么可以通过PC上的串口去执行命令?
-
1、uboot(负责初始化基本硬bai件,如串口,网卡,usb口等,然du后引导系统zhi运行)2、linux系统(真正的操作系统)3、你的应用程序(基于操作系统的软件应用)当你开发板上电时,u...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- go语言也可以做gui,go-fltk让你做出c++级别的桌面应用
- 旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗
- codeblocks和VS2019下的fltk使用中文
- FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库
- 中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux
- Linux 5.13内核有望合并对苹果M1处理器支持的初步代码
- Ubuntu系统下COM口测试教程(ubuntu port)
- 湖北嵌入式软件工程师培训怎么选,让自己脱颖而出
- 新阁上位机开发---10年工程师的Modbus总结
- 创建你的第一个可运行的嵌入式Linux系统-5
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)