专栏首页IT技能应用Spring Data JDBC - 如何使用自定义 ID 生成
原创

Spring Data JDBC - 如何使用自定义 ID 生成

原标题:Spring认证|Spring Data JDBC-如何使用自定义ID生成

这是关于如何解决使用 Spring Data JDBC 时可能遇到的各种挑战的系列文章的第一篇。

如果你不了解 Spring Data JDBC,你应该首先阅读它的介绍和文章,它解释了 Spring Data JDBC 上下文中的相关性。相信我,这很重要。

文章基于我在 2021 年春季一期上这篇文章的部分演讲。

使用 ID - 特别是当您想要控制实体的 ID 并且不会选择什么数据库时,您的选择是什么。

假设情况下,类型数据列JDBC假设的ID通过生成SERIAL或AUTOINCREMENT得到。 ,聚合根执行插入操作。数据库生成一个ID,这个ID由Spring Data JDBC在聚合根中设置。

考虑一个由单个简单的类组成的简单聚合:

类小黄人{

@ID

长ID;

字符串名称;

Minion(字符串名称){

this.name = 名称;

}

}

进一步考虑默认CrudRepository。

接口 MinionRepository 扩展 CrudRepository {

}

存储库会自动连接到您的代码中,如下所示:

@自动连线

MinionRepository 随从;

以下工作正常:

Minion before = new Minion("Bob");

assertThat(before.id).isNull();

Minion after = minions.save(before);

assertThat(after.id).isNotNull();

但是下一点点:

Minion before = new Minion("Stuart");

before.id = 42L;

minions.save(before);

更新语句,Spring Data JDBC 尝试执行更新,因为 ID 已经设置。但是,因为实际上是新的,更新语句影响零行 Spring Data JDBC 抛出异常。

有几种方法可以解决这个问题。我已经找到了你不同的解决方法,并且已经找到了我认为最简单的方法,因此可以找到适合的方法,你就可以停止阅读。之后回来阅读其他选项并提高您的 Spring Data 技能。

版本

将版本属性添加到您的聚合属性。“版本属性”是指用@Version。此类的主要目的是可以乐观锁定。但是,作为属性,Spring Data JDBC 使用版本属性来确定聚合根是否是新的。 只要版本是null 或0 原始类型,聚合就被认为是新的,即使id设置了。

使用这种方法,您必须更改实体和(当然)系统,但别无其他。

此外,对于许多应用程序来说,乐观的最初是很多。

我们把原来的Minion变成了一个VersionedMinion:

类 VersionedMinion {

@Id 长 ID;

字符串名称;

@Version 整数版本;

VersionedMinion(长ID,字符串名称){

this.id = id;

this.name = 名称;

}

}

通过此更改,以下构造有效:

VersionedMinion before = new VersionedMinion(23L, "Bob");

assertThat(before.id).isNotNull();

versionedMinions.save(before);

VersionedMinion 重新加载 = versionedMinions.findById(before.id).get();

assertThat(reloaded.name).isEqualTo("Bob");

样板

一种让您的遗赠附带 ID 的方法是自己另外插入物。您可以通过注入 JdbcAggregateTemplate 并调用 JdbcAggregateTemplate.insert(T)。这JdbAggregateTemplate是存储库下面的底层,因此您使用存储库用于插入的相同代码,但您决定何时使用插入:

Minion before = new Minion("Stuart");

before.id = 42L;

模板.插入(之前);

Minion reloaded = minions.findById(42L).get();

assertThat(reloaded.name).isEqualTo("Stuart");

请注意,我们不使用存储库农场使用模板,其中注入了以下内容:

@自动连线

JdbcAggregateTemplate 模板;

事件监听器

模板方法非常适用于您已经知道 ID 的情况 - 例如,当您从另一个系统导入数据并且您想要重用该系统的 ID 时。

如果您不知道 ID 并且不想在您的业务代码中包含任何 ID 相关的内容,那么使用 ID 可能是更好的选择。

我们的目的正确的目的是在某些生命周期事件期间被调用的豆子。它返回修改潜在的聚合根,因此它也适用于不形成实体类。

在目标中,我们确定有问题的聚合根是否需要新 ID。 如果是这样,我们将使用我们选择的算法生成它。

我们使用另一种变体 Minion

类 StringIdMinion {

@ID

字符串标识;

字符串名称;

StringIdMinion(字符串名称){

this.name = 名称;

}

}

但是,我们在配置中注册了一个惊人的例子:

@豆角,扁豆

BeforeSaveCallback beforeSaveCallback() {

返回(minion,mutableAggregateChange)-> {

如果(minion.id == null){

minion.id = UUID.randomUUID().toString();

}

返回仆从;

};

}

保存实体的代码现在看起来就像是由数据库生成的:

StringIdMinion before = new StringIdMinion("Kevin");

stringions.save(before);

assertThat(before.id).isNotNull();

StringIdMinion reloaded = stringions.findById(before.id).get();

assertThat(reloaded.name).isEqualTo("Kevin");

持久的

一个选项是让化根控制是否应该更新或插入。你可以实现持久化的方法(尤其是实现是新的)来实现这一点。您也想使用聚合根进行更新时,这会抓住。在这种情况下,您需要提出更灵活的策略。

我们需要 Minion 再次调整我们的:

类 PersistableMinion 实现 Persistable {

@Id 长 ID;

字符串名称;

PersistableMinion(长ID,字符串名称){

this.id = id;

this.name = 名称;

}

@覆盖

公共长 getId() {

返回标识;

}

@覆盖

公共布尔 isNew() {

// 这个实现肯定不适合生产使用

返回真;

}

}

保存一个的代码 PersistableMinion 看起来是一样的:

PersistableMinion before = new PersistableMinion(23L, "Dave");

persistableMinions.save(before);

PersistableMinion 重新加载 = persistableMinions.findById(before.id).get();

assertThat(reloaded.name).isEqualTo("Dave");

结论

Spring Data JDBC 提供了大量关于如何控制聚合 ID 的选项。虽然我在示例中使用了非常严重的逻辑,但基本没有什么能阻止您实现您所考虑的任何逻辑,因为它们都归结为 Java 代码。

完整的示例代码可在Spring中国教育管理中心(Spring认证)数据示例库访问!

原创声明,本文系作者授权云+社区发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 听说过spring-data-jdbc么?来个最佳实践

    很多人知道Mybatis,知道Jpa,但对2019年新诞生的一门技术知之甚少。那就是:spring-data-jdbc。这个标题起的很普通,但是内容绝对是最新的...

    xjjdog
  • 分布式秒杀实战之订单数据分表

    一般来说电商的日订单都是百千万级甚至是亿万级别的了,小小的数据库肯定是撑不住的,这时候就要提前考虑分库分表了。

    小柒2012
  • Spring Data JDBC、引用和聚合

    之前的博客文章中,我、描述了如何设置和使用 Spring Data JDBC。我还描述了使 Spring Data

    IT胶囊
  • SpringBoot入门建站全系列(二十四)使用Sharding-JDBC进行分库分表

    一个系统最初的线上业务量并不会很大,比如说单库的数据量在百万级别以下(事实上千万级别以下都还能支撑),那么MySQL的单库即可完成任何增/删/改/查的业务操作。...

    品茗IT
  • SpringBoot入门建站全系列(二十四)使用Sharding-JDBC进行分库分表

    一个系统最初的线上业务量并不会很大,比如说单库的数据量在百万级别以下(事实上千万级别以下都还能支撑),那么MySQL的单库即可完成任何增/删/改/查的业务操作。...

    品茗IT
  • 盘点 Java 数据库访问框架——究竟哪个更适合你

    假设您正在开发一个Java程序,有许多办法可以让您的应用连上数据库。下面会列举各数据库访问框架的适用场景,相信能够帮您选到适合项目的开发框架。

    PHP开发工程师
  • Spring整合Sharding-JDBC分库分表详情

    最初线上系统的业务量不是很大,业务数据量并不大,比如说单库的数据量在百万级别以下(事实上千万级别以下都还能支撑),那么MySQL的单库即可完成任何增/删/改/查...

    品茗IT
  • ShardingSphere解决海量数据分库分表

    互联网高速发展,同时也带来的海量数据存储问题。传统关系型数据库的单库单表已经很难支撑,如何高效存储和访问这些数据,成为业内急需解决的问题。解决思路有两个方向:

    用户7676729
  • SpringBoot 数据篇之使用JDBC

    Spring Data 包含对 JDBC 的存储库支持,并将自动为 CrudRepository 上的方法生成 SQL。对于更高级的查询,提供了 @Query ...

    静默虚空
  • [SpingBoot guides系列翻译]通过JDBC和Spring访问关系数据库

    build的时候用到了spring-boot-maven-plugin插件。他提供了很多便捷的特性。

    _淡定_
  • 芋道 Spring Boot JPA 入门(一)之快速入门

    摘要: 原创出处 http://www.iocoder.cn/Spring-Boot/JPA/ 「芋道源码」欢迎转载,保留摘要,谢谢!

    芋道源码
  • Springboot 系列(十)使用 Spring data jpa 访问数据库

    Springboot data jpa 和 Spring jdbc 同属于 Spring开源组织,在 Spring jdbc 之后又开发了持久层框架,很明显 S...

    未读代码
  • 数据量大了一定要分表,分库分表Sharding-JDBC入门与项目实战

    最近项目中不少表的数据量越来越大,并且导致了一些数据库的性能问题。因此想借助一些分库分表的中间件,实现自动化分库分表实现。调研下来,发现Sharding-JDB...

    程序员白楠楠
  • spring data jpa 使用自定义repository实现类

    路过君
  • What?数据量巨大还不分库分表?JDBC 入门与项目实战

    最近项目中不少表的数据量越来越大,并且导致了一些数据库的性能问题。因此想借助一些分库分表的中间件,实现自动化分库分表实现。调研下来,发现Sharding-JDB...

    用户2781897
  • ORM和 Spring Data Jpa

    什么是“持久化” 持久化的主要应用是将内存中的数据存储在关系型的数据库中,当然也可以存储在磁盘文件中、XML数据文件中等等。

    chenchenchen
  • ApiBoot - ApiBoot Security Oauth 依赖使用文档

    ApiBoot是一款基于SpringBoot1.x,2.x的接口服务集成基础框架, 内部提供了框架的封装集成、使用扩展、自动化完成配置,让接口开...

    恒宇少年
  • Spring Data(一)概念和仓库的定义

    Spring Data的主要任务是为数据访问提供一个相似的、一致的、基于Spring的编程模型,同时又保留着下面各个数据存储的特征。它使得使用数据访问技术非常的...

    小忽悠
  • springboot之整合基本的jdbc并操作Mysql数据库

    对于数据访问层,无论是SQL还是NOSQL,springboot默认采用整合spring data方式进行统一处理,添加大量自动配置,屏蔽了许多设置,引入各种x...

    西西嘛呦

扫码关注云+社区

领取腾讯云代金券