JPA/Hibernate嵌入式复合主键处理Null ID生成错误的最佳实践

jpa/hibernate嵌入式复合主键处理null id生成错误的最佳实践

本文旨在解决JPA/Hibernate中使用`@EmbeddedId`作为复合主键时,因外键关联未正确嵌入导致`Null ID`生成错误的问题。通过将`@ManyToOne`关联直接整合到`@Embeddable`类中,并优化实体映射与保存逻辑,确保复合主键在持久化前完整初始化,从而避免运行时错误,提升数据模型的一致性和健壮性。

理解JPA/Hibernate中嵌入式复合主键的挑战

在使用JPA和Hibernate构建数据模型时,复合主键是一种常见需求,尤其当一个实体的主键由多个字段组成时。@EmbeddedId注解允许我们将一个独立的@Embeddable类用作实体的主键。然而,当这个复合主键的一部分是一个外键(即关联到另一个实体的主键)时,如果没有正确配置,很容易遇到“Null ID generated”错误。

问题的核心在于,当一个实体(例如BlockAttribute)使用@EmbeddedId,并且该@EmbeddedId包含一个外键(例如blockID,指向Block实体的主键),在保存BlockAttribute之前,BlockAttributeID中的所有组件都必须被正确初始化。如果BlockAttributeID仅仅包含一个Long blockID字段,而BlockAttribute实体本身又有一个@ManyToOne Block block字段,那么在保存BlockAttribute时,JPA/Hibernate可能无法自动将Block实体的主键值填充到BlockAttributeID中的blockID字段。

考虑以下初始的数据模型:

1. BlockAttributeID (嵌入式主键类)

@Embeddable@Data // Lombok注解,用于生成getter/setter, equals, hashCode等public class BlockAttributeID implements Serializable {    @Column(name = "block_id")    Long blockID; // 仅包含Block的ID    String attribute;    // equals 和 hashCode 方法的实现需要注意,尤其是当blockID可能为null时    @Override    public boolean equals(Object o) {        if (this == o) return true;        if (!(o instanceof BlockAttributeID)) return false;        BlockAttributeID that = (BlockAttributeID) o;        return Objects.equals(blockID, that.blockID) && Objects.equals(attribute, that.attribute);    }    @Override    public int hashCode() {        return Objects.hash(blockID, attribute);    }}

2. BlockAttribute (使用嵌入式主键的实体)

@Data@Table(name = "block_attribute")@Entitypublic class BlockAttribute {    @EmbeddedId    BlockAttributeID blockAttributeID;    // 冗余的ManyToOne关联,与EmbeddedId中的blockID形成冲突或混淆    @ManyToOne(fetch = FetchType.LAZY)    @JsonIgnore    @JoinColumn(name = "block_id") // 这个@JoinColumn通常会导致问题    Block block; // 这里又有一个Block实体引用    String label;    // ... 其他字段    // equals 和 hashCode 同样需要基于复合主键进行正确实现    @Override    public boolean equals(Object o) {        if (this == o) return true;        if (!(o instanceof BlockAttribute)) return false;        BlockAttribute that = (BlockAttribute) o;        return Objects.equals(blockAttributeID, that.blockAttributeID);    }    @Override    public int hashCode() {        return Objects.hash(blockAttributeID);    }}

3. Block (父实体)

@Table(name = "block")@Entity@Datapublic class Block {    @Id    @GeneratedValue(strategy = GenerationType.IDENTITY)    @Column(name = "block_id")    Long blockID; // Block的主键    // ... 其他字段和关联    @OneToMany(mappedBy = "block", fetch = FetchType.LAZY, cascade = CascadeType.ALL, orphanRemoval = true)    Set blockAttributes = new HashSet();    // ... 其他方法}

当尝试以下保存逻辑时,就会出现Null ID generated for: class BlockAttribute错误:

// 1. 保存父Block实体,生成其blockIDblock = blockRepository.save(block);// 2. 设置BlockAttribute的block字段blockAttribute.setBlock(block); // 此时blockAttributeID中的blockID并未被设置// 3. 尝试保存BlockAttributeblockAttributeRepository.save(blockAttribute); // 抛出Null ID错误

问题在于,blockAttribute.setBlock(block)只是设置了BlockAttribute实体中的block引用,但@EmbeddedId中的blockID字段仍然是null。JPA在保存BlockAttribute时,需要BlockAttributeID中的所有主键组件都非空。

解决方案:将外键关联嵌入到@Embeddable类中

解决此问题的关键在于,如果一个外键是复合主键的一部分,那么该外键的@ManyToOne关联应该直接放在@Embeddable类中,而不是在主实体中重复定义。这样,@EmbeddedId就能直接持有对关联实体的引用,从而确保在创建复合主键时,能够获取到关联实体的主键信息。

Ai Mailer Ai Mailer

使用Ai Mailer轻松制作电子邮件

Ai Mailer 49 查看详情 Ai Mailer

1. 修正后的 BlockAttributeID 类

我们将@ManyToOne关联直接移入BlockAttributeID。

import com.fasterxml.jackson.annotation.JsonIgnore;import lombok.Data; // 推荐使用Lombok简化代码import javax.persistence.*;import java.io.Serializable;import java.util.Objects;@Embeddable@Data // 确保生成了getter/setter以及默认的equals/hashCode,但需手动优化public class BlockAttributeID implements Serializable {    // 将ManyToOne关联直接嵌入到复合主键类中    @ManyToOne(fetch = FetchType.LAZY)    @JsonIgnore // 通常在嵌入式ID中,避免序列化Block实体,防止循环引用    @JoinColumn(name = "block_id", referencedColumnName = "block_id") // 明确指定关联列    Block block; // 现在直接持有Block实体引用    String attribute;    // 构造函数,方便创建复合主键实例    public BlockAttributeID(Block block, String attribute) {        this.block = block;        this.attribute = attribute;    }    // JPA规范要求存在无参构造函数    public BlockAttributeID() {    }    // 优化后的equals方法:基于Block的ID和attribute进行比较    @Override    public boolean equals(Object o) {        if (this == o) return true;        if (!(o instanceof BlockAttributeID)) return false;        BlockAttributeID that = (BlockAttributeID) o;        // 比较Block实体时,应比较其主键ID,而不是整个实体对象,以避免代理问题        return Objects.equals(            this.block != null ? this.block.getBlockID() : null,            that.block != null ? that.block.getBlockID() : null        ) && Objects.equals(this.attribute, that.attribute);    }    // 优化后的hashCode方法:基于Block的ID和attribute生成    @Override    public int hashCode() {        return Objects.hash(            this.block != null ? this.block.getBlockID() : null,            this.attribute        );    }}

关键点:

@ManyToOne Block block; 直接定义在BlockAttributeID中。@JoinColumn(name = “block_id”, referencedColumnName = “block_id”) 明确指定了外键列。equals() 和 hashCode() 方法被优化,以Block的ID和attribute字段作为比较和哈希的依据,这对于包含实体引用的@Embeddable类至关重要。

2. 修正后的 BlockAttribute 类

由于BlockAttributeID现在已经包含了Block的关联信息,BlockAttribute实体中的冗余@ManyToOne Block block;字段应该被移除。

import lombok.Data;import javax.persistence.*;import java.util.Objects;@Data@Table(name = "block_attribute")@Entitypublic class BlockAttribute {    @EmbeddedId    BlockAttributeID blockAttributeID; // 复合主键,现在包含了Block的引用    // 移除冗余的Block字段,因为它已经包含在blockAttributeID中    // @ManyToOne(fetch = FetchType.LAZY)    // @JsonIgnore    // @JoinColumn(name = "block_id")    // Block block;    String label;    @Enumerated(EnumType.STRING)    Type type;    @Enumerated(EnumType.STRING)    Unit unit;    String value;    // equals 和 hashCode 应该基于 @EmbeddedId    @Override    public boolean equals(Object o) {        if (this == o) return true;        if (!(o instanceof BlockAttribute)) return false;        BlockAttribute that = (BlockAttribute) o;        return Objects.equals(blockAttributeID, that.blockAttributeID);    }    @Override    public int hashCode() {        return Objects.hash(blockAttributeID);    }}

关键点:

移除了BlockAttribute中直接的@ManyToOne Block block字段,避免了重复映射和潜在的混淆。equals()和hashCode()现在完全依赖于blockAttributeID,确保了一致性。

修正后的保存逻辑

在实体映射调整后,保存逻辑也需要相应修改,以确保在创建BlockAttribute时,其@EmbeddedId能够被正确初始化。

// 1. 首先保存父Block实体,确保其主键(blockID)已生成Block savedBlock = blockRepository.save(block);// 2. 创建BlockAttributeID实例,传入已保存的Block实体和attribute值// 此时savedBlock已经拥有了数据库生成的主键IDBlockAttributeID blockAttributeID = new BlockAttributeID(savedBlock, completeBlockDTO.getBlockAttributeDTO().getAttribute());// 3. 将创建好的BlockAttributeID设置到BlockAttribute实体中blockAttribute.setBlockAttributeID(blockAttributeID);// 4. 保存BlockAttribute实体blockAttributeRepository.save(blockAttribute);// 对于其他依赖于Block的子实体(如BlockBoundary),如果其关联方式是ManyToOne,// 则可以直接设置Block实体引用,因为它的ID是独立的,不作为其复合主键的一部分。// blockBoundary.setBlock(savedBlock);// blockBoundaryRepository.save(blockBoundary);

总结与最佳实践

外键作为复合主键的一部分: 当一个外键是@EmbeddedId的一部分时,应将@ManyToOne关联直接定义在@Embeddable类中,而不是在主实体中重复定义。@Embeddable中的equals()和hashCode(): 务必为@Embeddable类正确实现equals()和hashCode()方法。如果@Embeddable类包含实体引用(如Block block),则在这些方法中应比较关联实体的主键ID,而不是整个实体对象,以避免Hibernate代理对象带来的问题。保存顺序: 在保存使用@EmbeddedId的子实体之前,必须先保存其关联的父实体,以确保父实体的主键已经生成并可用于构建复合主键。避免冗余映射: 如果外键关联已在@EmbeddedId中定义,则主实体中不应再有重复的@ManyToOne映射到同一外键,这可能导致混淆或错误。@JoinColumn的精确性: 在@ManyToOne映射中使用@JoinColumn时,确保name和referencedColumnName属性准确无误,指向正确的数据库列。

遵循这些最佳实践,可以有效避免JPA/Hibernate中嵌入式复合主键相关的Null ID生成错误,构建出更加健壮和易于维护的数据模型。

以上就是JPA/Hibernate嵌入式复合主键处理Null ID生成错误的最佳实践的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1074049.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月2日 07:01:32
下一篇 2025年12月2日 07:01:53

相关推荐

  • 如何让 PHP 处理 XML 和 JSON 数据?

    php 可轻松处理 xml 和 json 数据。使用 simplexml 扩展处理 xml 数据,包括加载文档、访问元素和遍历元素。使用 json_decode() 函数处理 json 数据,包括转换字符串、访问属性和遍历数组。实战案例包括使用 php 解析天气 api 的 json 数据。 使用 …

    2025年12月9日
    000
  • 如何用 PHP 调用 Java 函数?

    使用 java bridge 类库可从 php 脚本中调用 java 函数,通过以下步骤实现:使用 composer 安装 java bridge 类库。使用 setjavaclasspath() 方法配置 php 代码和 java 类路径之间的链接。使用 javaclass::callstatic…

    2025年12月9日
    000
  • 使用第三方 PHP 函数扩展应用程序功能

    第三方 php 函数通过 composer 安装后,可以通过 psr-4 自动加载。它们可用于扩展应用程序功能,例如使用 guzzle 进行 http 请求或使用 emailvalidator 验证电子邮件地址。通过利用第三方函数,开发人员可以轻松地在应用程序中添加新功能,而无需重新编写代码。 使用…

    2025年12月9日
    000
  • 使用第三方 PHP 函数时避免常见陷阱

    使用第三方 php 函数时,必须注意陷阱,包括:确保依赖关系明确,检查函数签名,处理错误,验证结果。这些准则可避免错误和意外行为,确保代码的可靠性和健壮性。实时案例:使用 guzzlehttp 时,请记住将响应对象转换为字符串或数组,以避免常见陷阱。 使用第三方 PHP 函数时避免常见陷阱 在使用第…

    2025年12月9日
    000
  • PHP 引用传递:加速你的函数开发流程

    引用传递允许函数通过修改变量引用来修改其参数的原始值,从而提高函数的效率,尤其适用于处理大型或复杂数据结构。语法为在参数前面加上”&”符号;实战案例中,通过引用传递数组,可以修改原始数组,而非仅打印副本。 PHP 引用传递:加速你的函数开发流程 引用传递允许函数修改其…

    2025年12月9日
    000
  • PHP 递归函数在解析 JSON 数据中的作用

    递归函数在解析 json 数据中发挥着关键作用,通过使用 json_decode() 函数和递归算法,可以有效解析复杂嵌套的数据结构:定义递归函数 parsejson(),将 json 字符串作为输入。对 json 结果中的每个键值对进行迭代。如果值是数组,则递归调用 parsejson(),将数组…

    2025年12月9日
    000
  • PHP 函数怎么处理 JSON 数据

    php 提供了处理 json 数据的函数,使开发人员能够轻松地编码和解码 json 数据。1. 编码 json 数据:json_encode() 将 php 变量编码为 json 字符串。2. 解码 json 数据:json_decode() 将 json 字符串解码为 php 变量。实战案例中,可…

    2025年12月9日
    000
  • PHP 函数如何与 Java 交互

    php 函数可以通过以下步骤与 java 交互:包含 java 类创建 java 对象调用 java 方法访问 java 字段创建数组设置数组元素を活用例としては、java で数字の合計を計算するクラスを作成し、php スクリプトからこのクラスを使用して計算を実行できます。 PHP 函数如何与 Ja…

    2025年12月9日
    000
  • PHP 函数如何返回 JSON

    php 通过 json_encode() 函数将数据转换为 json 格式,它接受变量作为参数并返回 json 字符串。使用 json_encode() 函数和设置适当的 http 头,你可以轻松地将 json 数据作为响应返回给客户端,或使用其他选项,例如 json_decode() 和 json…

    2025年12月9日
    000
  • PHP 函数如何与 JSON 交互

    php 函数在处理 json 数据时,提供以下功能:json_encode() 将 php 变量转换为 json 字符串。json_decode() 将 json 字符串解析为 php 变量。具体语法和实战案例见文章。 PHP 函数如何与 JSON 交互 JSON 是一种广泛用于在 Web 应用程序…

    2025年12月9日
    000
  • PHP 函数名称中的缩写规则

    在 php 函数命名中,缩写应遵循以下规则:1. 相同含义的缩写保持一致;2. 缩写易于理解;3. 缩写尽可能短;4. 主要单词不缩写。通过遵循这些规则,可创建更清晰的 php 函数。 PHP 函数名称中的缩写规则 在 PHP 函数命名中,缩写是常见的做法,可以帮助函数名称更简洁、表达更明确。以下是…

    2025年12月9日
    000
  • PHP 函数如何获取 JSON 数据?

    php 中获取 json 数据的方法:使用 json_decode() 函数将 json 字符串解码为 php 变量。使用 file_get_contents() 函数从 url 获取 json 数据,再使用 json_decode() 函数解析。使用 json_encode() 函数将 php 变…

    2025年12月9日
    000
  • PHP 函数名称中允许使用的字符

    php 函数名称中允许字母、数字和下划线,不允许空格和特殊字符(除下划线外)。命名约定包括:以小写字母或下划线开头,使用驼峰命名法,避免与内置函数或变量冲突。 PHP 函数名称中允许使用的字符 PHP 函数名称中允许使用的字符遵循严格的规则,如下: 允许的字符: 立即学习“PHP免费学习笔记(深入)…

    2025年12月9日
    000
  • PHP 变量和函数命名的区别

    php 中变量和函数命名方式不同:变量以 $ 符号开头,使用驼峰或下划线命名法,描述性强;函数不以 $ 符号开头,仅用驼峰命名法,表示其功能。 PHP 变量和函数命名的区别 在 PHP 中,变量和函数的命名规则截然不同。理解这些差异对于编写整洁、可读性高的代码至关重要。 变量命名 立即学习“PHP免…

    2025年12月9日
    000
  • PHP框架社区的活跃程度对比

    在 php 框架中,社区活跃程度的衡量指标包括贡献者数量、问题的响应时间和支持的文档。laravel 拥有最活跃的社区,其丰富的贡献者、快速的响应时间和全面的文档使其成为初学者和经验丰富的开发人员的理想选择。symfony 提供稳定性,而 codeigniter 以易用的文档吸引初学者。 PHP 框…

    2025年12月9日
    000
  • 币安交易所(binance)新手如何进行合约交易操作及防爆仓指南

    币安合约交易需先熟悉界面,包括交易对、K线图、委托区和仓位信息,重点关注强平价格;执行交易时选择交易对、设置杠杆(新手建议低倍)、下单类型及数量,确认后提交;开仓后应设置止盈止损以控制风险;逐仓模式下可追加保证金降低强平风险;根据风险偏好在全仓与逐仓间切换保证金模式,全仓风险更高但资金利用率高。 币…

    2025年12月9日
    000
  • 以太坊是公链吗还是私链?大白话讲解

    很多刚接触区块链的朋友都会好奇,以太坊到底是公有的还是私有的?这篇文章将用最简单直白的方式,帮您彻底搞懂公链和私链的区别,并明确以太坊的真正属性。 一、什么是公链? 1、公链,全称公共区块链,顾名思义,它是一个对全世界开放的网络。 2、任何人都可以自由地加入这个网络,读取数据、发送交易,并参与到交易…

    2025年12月9日
    000
  • 详解灵魂绑定代币(SBT),它如何构建Web3世界的数字身份?

    SBT是不可转让的数字凭证,用于记录Web3中的身份、成就与社会关系;它由发行者铸造并绑定至用户地址,确保身份真实唯一,防止冒用与虚假信誉;通过叠加教育、工作、贡献等多维数据,SBT为匿名地址赋予人格,构建去中心化身份体系;大学、开源项目、DAO等可发放学位、参与证明及治理权限类SBT,用户聚合形成…

    2025年12月9日
    000
  • 币安交易所全球官网入口 Binance官方认证APP下载地址

    币安 binance 是全球领先的数字资产交易平台,支持现货交易、期货合约、理财产品等多样化服务。本文将为你介绍 币安全球官网入口 及其 官方 app 下载方式,帮助你安全访问并安装。 币安全球官网入口 建议通过官方域名访问币安官网:— 在这里你可以完成注册、登录、资产管理、充值提现等操作。 币安官…

    2025年12月9日
    000
  • 一文了解币圈:以太坊在哪里诞生?怎么购买?有什么作用?

    以太坊(ethereum)作为区块链技术的重要里程碑,不仅是一种数字资产,更是一个支持去中心化应用的全球性开源平台。本文旨在为初学者快速梳理以太坊的起源、获取方式及其核心应用场景,帮助您构建一个清晰的认知框架。 一、以太坊的诞生 1、以太坊最初由程序员 Vitalik Buterin 在2013年提…

    2025年12月9日
    000

发表回复

登录后才能评论
关注微信