百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

多年的教训:根据DDD设计原则改变JPA/Hibernate的使用方式

liuian 2025-07-08 20:04 19 浏览

我(lorenzo)最近一直在更新一些培训材料,思考JPA更好的教学方法和讨论方式。我一直在思考的一件事是我们通常是如何使用JPA?这里结合我所经历的(和观察到的)痛苦,应该如何改变传统使用方式?

JPA通常被视为一组注释(或XML文件),它们提供O/R(对象关系)映射信息。大多数开发人员认为他们知道和使用的映射注释越多,他们得到的好处就越多。但是在过去的几年里,与中小规模的巨石/单体/整体系统(大约有200张表/实体)的搏斗教会了我一些别的东西。

教训:

  • 按ID引用实体(仅映射聚合中的实体关系)
  • 不要让JPA窃取你的ID(尽可能避免@GeneratedValue)
  • 使用特殊join来join不相关的实体

按标识符ID引用实体

仅映射DDD聚合中的实体关系。

传统 JPA或Hibernate教程(和培训)通常会涵盖所有可能的实体关系映射。在教学基本映射之后,许多映射将从简单的单向@manytone映射开始。然后继续双向@OneToMany和@ManyToOne。不幸的是,大多数情况下,他们没有明确指出,这种映射关系不是很好。因此,初学者在完成训练时往往会认为,不映射相关实体是错误的。他们错误地认为外键字段必须映射为相关实体。

@Entity
public class SomeEntity {
    // ...
    @ManyToOne private Country country;
    // ...
}
 
@Entity
public class Country {
    @Id private String id; // e.g. US, JP, CN, CA, GB, PH
    // ...
}

将上面@ManyToOne 应该改为@Column,将相关实体的主键映射为一个字段即可:

@Entity
public class SomeEntity {
    // ...
    @Column private String countryId;
    // ...
}
 
@Entity
public class Country {
    @Id private String id; // e.g. US, JP, CN, CA, GB, PH
    // ...
}

映射所有实体关系会增加了不必要的遍历的机会,这通常会导致不必要的内存消耗。这也会导致不必要的EntityManager操作级联。

如果您只处理少数几个实体/表,这可能并不多。但是当与几十个(如果不是几百个)实体一起工作时,它就变成了维护的噩梦。

何时映射相关实体?

仅当相关实体位于聚合中时才映射它们(在DDD中)。

聚合是领域驱动设计中的一种模式。DDD聚合是可以作为单个单元处理的域对象的集群。例如订单及其行项目,它们将是单独的对象,但将订单(及其行项目)视为单个聚合非常有用。

https://martinfowler.com/bliki/DDD_Aggregate.html

@Entity
public class Order {
    // ...
    @OneToMany(mappedBy = "order", ...) private List<OrderItem> items;
    // ...
}
 
@Entity
public class OrderItem {
    // ...
    @ManyToOne(optional = false) private Order order;
    // ...
}

更现代的聚合设计方法提倡在聚合之间进行更干净的分离。通过存储聚合根的ID(唯一标识符)而不是完整的引用来引用聚合根是一种很好的做法。

如果我们展开上面的简单订单示例,那么行项目(OrderItem类)不应该有到产品的@ManyToOne映射,相反,它应该只有产品的ID:

@Entity
public class Order {
    // ...
    @OneToMany(mappedBy = "order", ...) private List<OrderItem> items;
    // ...
}
 
@Entity
public class OrderItem {
    // ...
    @ManyToOne(optional = false) private Order order;
    // @ManyToOne private Product product; // <-- Avoid this!
    @Column private ... productId;
    // ...
}

但是…如果产品(聚合根实体)的@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满意。

@Entity
public class Order {
    @Id private Long id;
    // ...
    public Order(Long id) {
        // ...
        this.id = id;
    }
    public Long getId() { return id; }
    // ...
    protected Order() { /* as required by ORM/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)。

public class SimpleJpaRepository // ...
    @Override
    public <S extends T> save(S entity) {
        // ...
        if (entityInformation.isNew(entity)) {
            em.persist(entity);
            return entity;
        } else {
            return em.merge(entity);
        }
    }
}

精明的读者会注意到,如果@Id字段从不为null,save()方法将始终调用merge()。这会导致不必要的SQL SELECT(在预期的INSERT之前)。

幸运的是,解决方法很简单-实现 Persistable<ID>.

@MappedSuperclass
public abstract class BaseEntity<ID> implements Persistable<ID> {
    @Transient
    private boolean persisted = false;
    @Override
    public boolean isNew() {
        return !persisted;
    }
    @PostPersist
    @PostLoad
    protected void setPersisted() {
        this.persisted = true;
    }
}
以上还意味着对实体的所有更新都必须首先将现有实体加载到持久性上下文中,然后将更改应用到托管实体。

使用特殊Join连接来join不相关的实体

那么,连接join呢?既然我们通过ID引用了其他实体,那么如何在JPA中连接join不相关的实体呢?

在jpa2.2版本中,不相关的实体不能连接。但是,我无法确认这是否已经成为3.0版的标准,在3.0版中,所有javax.persistence引用都被重命名为jakarta.persistence。

给定OrderItem实体,缺少@manytone映射会导致它无法与产品实体联接。

@Entity
public class Order {
    // ...
}
 
@Entity
public class OrderItem {
    // ...
    @ManyToOne(optional = false) private Order order;
    @Column private ... productId;
    // ...
}

值得庆幸的是,Hibernate 5.1.0+(2016年发布)和EclipseLink 2.4.0+(2012年发布)一直在支持无关实体的连接。这些连接也称为特殊连接 ad-hoc joins。

SELECT o
  FROM Order o
  JOIN o.items oi
  JOIN Product p ON (p.id = oi.productId) -- supported in Hibernate and EclipseLink

另外,这也是一个API问题(支持两个根实体的JOIN/ON)。我真的希望它能很快成为一种标准。

相关推荐

教你把多个视频合并成一个视频的方法

一.情况介绍当你有一个m3u8文件和一个目录,目录中有连续的视频片段,这些片段可以连成一段完整的视频。m3u8文件打开后像这样:m3u8文件,可以理解为播放列表,里面是播放视频片段的顺序。视频片段像这...

零代码编程:用kimichat合并一个文件夹下的多个文件

一个文件夹里面有很多个srt字幕文件,如何借助kimichat来自动批量合并呢?在kimichat对话框中输入提示词:你是一个Python编程专家,完成如下的编程任务:这个文件夹:D:\downloa...

Java APT_java APT 生成代码

JavaAPT(AnnotationProcessingTool)是一种在Java编译阶段处理注解的工具。APT会在编译阶段扫描源代码中的注解,并根据这些注解生成代码、资源文件或其他输出,...

Unit Runtime:一键运行 AI 生成的代码,或许将成为你的复制 + 粘贴神器

在我们构建了UnitMesh架构之后,以及对应的demo之后,便着手于实现UnitMesh架构。于是,我们就继续开始UnitRuntime,以用于直接运行AI生成的代码。PS:...

挣脱臃肿的枷锁:为什么说Vert.x是Java开发者手中的一柄利剑?

如果你是一名Java开发者,那么你的职业生涯几乎无法避开Spring。它如同一位德高望重的老国王,统治着企业级应用开发的大片疆土。SpringBoot的约定大于配置、SpringCloud的微服务...

五年后,谷歌还在全力以赴发展 Kotlin

作者|FredericLardinois译者|Sambodhi策划|Tina自2017年谷歌I/O全球开发者大会上,谷歌首次宣布将Kotlin(JetBrains开发的Ja...

kotlin和java开发哪个好,优缺点对比

Kotlin和Java都是常见的编程语言,它们有各自的优缺点。Kotlin的优点:简洁:Kotlin程序相对于Java程序更简洁,可以减少代码量。安全:Kotlin在类型系统和空值安全...

移动端架构模式全景解析:从MVC到MVVM,如何选择最佳设计方案?

掌握不同架构模式的精髓,是构建可维护、可测试且高效移动应用的关键。在移动应用开发中,选择合适的软件架构模式对项目的可维护性、可测试性和团队协作效率至关重要。随着应用复杂度的增加,一个良好的架构能够帮助...

颜值非常高的XShell替代工具Termora,不一样的使用体验!

Termora是一款面向开发者和运维人员的跨平台SSH终端与文件管理工具,支持Windows、macOS及Linux系统,通过一体化界面简化远程服务器管理流程。其核心定位是解决多平台环境下远程连接、文...

预处理的底层原理和预处理编译运行异常的解决方案

若文章对您有帮助,欢迎关注程序员小迷。助您在编程路上越走越好![Mac-10.7.1LionIntel-based]Q:预处理到底干了什么事情?A:预处理,顾名思义,预先做的处理。源代码中...

为“架构”再建个模:如何用代码描述软件架构?

在架构治理平台ArchGuard中,为了实现对架构的治理,我们需要代码+模型描述所要处理的内容和数据。所以,在ArchGuard中,我们有了代码的模型、依赖的模型、变更的模型等,剩下的两个...

深度解析:Google Gemma 3n —— 移动优先的轻量多模态大模型

2025年6月,Google正式发布了Gemma3n,这是一款能够在2GB内存环境下运行的轻量级多模态大模型。它延续了Gemma家族的开源基因,同时在架构设计上大幅优化,目标是让...

比分网开发技术栈与功能详解_比分网有哪些

一、核心功能模块一个基本的比分网通常包含以下模块:首页/总览实时比分看板:滚动展示所有正在进行的比赛,包含比分、比赛时间、红黄牌等关键信息。热门赛事/焦点战:突出显示重要的、关注度高的比赛。赛事导航...

设计模式之-生成器_一键生成设计

一、【概念定义】——“分步构建复杂对象,隐藏创建细节”生成器模式(BuilderPattern):一种“分步构建型”创建型设计模式,它将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建...

构建第一个 Kotlin Android 应用_kotlin简介

第一步:安装AndroidStudio(推荐IDE)AndroidStudio是官方推荐的Android开发集成开发环境(IDE),内置对Kotlin的完整支持。1.下载And...