jpa vue管理系统_在JPA 2.1中使用@Convert正确完成映射枚举
jpa vue管理系统
如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱。 使用enum
作为@Entity
的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它们。 它给了您2 + 1个选择:
托肖夫达林
@Enumerated(EnumType.ORDINAL)
(默认值)将使用Enum.ordinal()
映射enum
值。 基本上,第一个枚举值将在数据库列中映射为0
,第二个映射为1
,依此类推。这非常紧凑,在您想要修改枚举时非常有用。 在中间删除或增加值或重新排列它们将完全破坏现有记录。 哎哟! 更糟糕的是,单元测试和集成测试通常在干净的数据库上运行,因此它们不会发现旧数据之间的差异。@Enumerated(EnumType.STRING)
更安全,因为它存储enum
字符串表示形式。 现在,您可以安全地添加新值并移动它们。 但是,重命名Java代码中的enum
仍会破坏DB中的现有记录。 更重要的是,这种表示非常冗长,不必要地消耗了数据库资源。- 您也可以使用原始表示形式(例如single
char
或int
),并在@PostLoad
/@PrePersist
/@PreUpdate
事件中手动来回映射它。 从数据库角度来看,最灵活,最安全,但是非常难看。
幸运的是,几天前发布的Java Persistence API 2.1 ( JSR-388 )提供了 可插拔数据转换器的标准化机制。 这样的API以专有形式存在很久了,它并不是真正的火箭科学,但是将其作为JPA的一部分是一个很大的改进。 据我所知, Eclipselink是迄今为止唯一可用的JPA 2.1实现,因此我们将使用它进行一些实验。
我们将从作为“穷人的CRUD:jqGrid,REST,AJAX和Spring MVC集成在房子里”的一部分开发的示例Spring应用程序开始。 该版本没有持久性,因此我们将在由Eclipselink支持的Spring Data JPA之上添加薄DAO层。 到目前为止,只有实体是Book
:
@Entity
public class Book {@Id@GeneratedValue(strategy = IDENTITY)private Integer id;//...private Cover cover;//...
}
其中Cover
是enum
:
public enum Cover {PAPERBACK, HARDCOVER, DUST_JACKET}
ORDINAL
和STRING
都不是一个很好的选择。 前者是因为以任何方式重新排列前三个值都会破坏现有记录的加载。 后者太冗长。 这是JPA中的自定义转换器起作用的地方:
import javax.persistence.AttributeConverter;
import javax.persistence.Converter;@Converter
public class CoverConverter implements AttributeConverter<Cover, String> {@Overridepublic String convertToDatabaseColumn(Cover attribute) {switch (attribute) {case DUST_JACKET:return "D";case HARDCOVER:return "H";case PAPERBACK:return "P";default:throw new IllegalArgumentException("Unknown" + attribute);}}@Overridepublic Cover convertToEntityAttribute(String dbData) {switch (dbData) {case "D":return DUST_JACKET;case "H":return HARDCOVER;case "P":return PAPERBACK;default:throw new IllegalArgumentException("Unknown" + dbData);}}
}
好吧,亲爱的读者,我不会侮辱您,对此进行解释。 将枚举转换为将存储在关系数据库中的任何内容,反之亦然。 从理论上讲,如果使用以下声明,则JPA提供程序应自动应用转换器:
@Converter(autoApply = true
它对我不起作用。 此外,在@Entity
类中显式声明它们而不是@Enumerated
也不起作用:
import javax.persistence.Convert;//...@Convert(converter = CoverConverter.class)
private Cover cover;
导致异常:
Exception Description: The converter class [com.blogspot.nurkiewicz.CoverConverter]
specified on the mapping attribute [cover] from the class [com.blogspot.nurkiewicz.Book] was not found.
Please ensure the converter class name is correct and exists with the persistence unit definition.
错误或功能,我不得不在orm.xml
提到转换器:
<?xml version="1.0"?>
<entity-mappings xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/orm" version="2.1"><converter class="com.blogspot.nurkiewicz.CoverConverter"/>
</entity-mappings>
它飞! 我可以自由修改我的Cover
枚举(添加,重新排列,重命名),而不会影响现有记录。
我想与您分享的一个技巧与可维护性有关。 每次您有一段从或到enum
的代码映射时,请确保已对其进行了正确的测试。 我并不是说要手动测试每个可能的现有值。 为了确保新的enum
值在映射代码中得到反映,我进行了更多测试。 提示:如果添加新的enum
值,但是忘记从中添加映射代码,则下面的代码将失败(通过抛出IllegalArgumentException
):
for (Cover cover : Cover.values()) {new CoverConverter().convertToDatabaseColumn(cover);
}
JPA 2.1中的自定义转换器比我们所看到的有用得多。 如果将JPA与Scala结合使用,则可以使用@Converter
将数据库列直接映射到scala.math.BigDecimal
, scala.Option
或小写类。 在Java中,最终将有一种可移植的方式来映射Joda时间。 最后但并非最不重要的一点是,如果您喜欢(非常)强类型的域,则可能希望拥有PhoneNumber
类(带有isInternational()
, getCountryCode()
和自定义验证逻辑),而不是String
或long
。 JPA 2.1中的这一小增加肯定会显着提高域对象的质量。
如果您想使用此功能,可以在GitHub上找到示例Spring Web应用程序。
翻译自: https://www.javacodegeeks.com/2013/06/mapping-enums-done-right-with-convert-in-jpa-2-1.html
jpa vue管理系统
jpa vue管理系统_在JPA 2.1中使用@Convert正确完成映射枚举相关推荐
- 1.0jpa 2.0_在JPA 2.1中使用@Convert正确完成映射枚举
1.0jpa 2.0 如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱. 使用enum作为@Entity的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它们 ...
- 在JPA 2.1中使用@Convert正确完成映射枚举
如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱. 使用enum作为@Entity的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它们. 它给了您2 + 1 ...
- jpa vue管理系统_如何通过利用Java流获取类型安全和直观的Hibernate / JPA查询
jpa vue管理系统 大部分Java数据库应用程序都在使用Hibernate / JPA来弥合Java和SQL之间的鸿沟. 直到最近,我们还被迫将Java和JPQL混合使用,或者使用复杂的命令式标准 ...
- jpa 查询 列表_终极JPA查询和技巧列表–第3部分
jpa 查询 列表 在阅读第三部分之前,请记住本系列的第一部分和第二部分 JPA:通过查询创建对象 JPA允许我们使用所需的值在查询内创建对象: package com.model;public cl ...
- jpa 查询 列表_终极JPA查询和技巧列表–第2部分
jpa 查询 列表 这一部分是该系列文章的第一部分 . JPA:NamedQuery,使用日期查询,有关getSingleResult方法的警告 为了避免重复查询代码,提高性能并简化维护查询,我们可以 ...
- jpa 查询 列表_终极JPA查询和技巧列表–第1部分
jpa 查询 列表 我们可以在Internet上找到一些JPA"如何做",在本博客的此处,教您如何使用JPA执行多项任务. 通常,我看到有人问有关使用JPA进行查询的问题. 通常, ...
- mysql jpa 批注 视图_通过JPA注解映射视图的实体类 jpa 视图 无主键 @Query注解的用法(Spring Data JPA) jpa 使用sql语句...
参考: https://blog.csdn.net/qq465235530/article/details/68064074 https://www.cnblogs.com/zj0208/p/6008 ...
- 将输出结果以json类型打印在控制台上_系列文章:Kubernetes中日志的正确输出姿势...
前言 上一期主要和大家介绍从全局维度考虑如何去构建K8s中的日志系统,本期我们从实践角度出发来一步步构建K8s中的日志监控体系.构建日志系统的第一步是如何去产生这些日志,而这也往往是最繁杂最困难的一步 ...
- 歌曲信息管理系统c语言代码,歌曲信息管理系统_通用完整版.doc
歌曲信息管理系统_通用完整版 PAGE 中 国 地 质 大 学 本科生课程论文封面 C语言课程设计 课程名称 计算机高级语言课程设计(C) 教师姓名 刘 刚 本科生姓名 本科生学号 本科生专业 机械设 ...
最新文章
- 最近的裁员事件告诉我们,录音笔太重要了!送10个给大家备用!
- 无法启动程序,拒绝访问解决方法
- linux 内核 netfilter 网络过滤模块 (4)-期望连接
- Flomaster 2020中文版
- c语言字符串加减_C语言中指针的介绍
- VTK:PolyData之TriangleSolidColor
- dede config.chche.inc.php,dede/config.php · 辉辉菜/三强源码 - Gitee.com
- TOMCAT启动到一半停止如何解决
- XCode、Objective-C、Cocoa 说的是几样东西
- pygame写游戏,常用代码记录
- 飞秋教程:日程安排记事提醒
- Qt:Qt实现飞秋拦截助手—ARP攻击
- 敏捷开发系列学习总结(1)——版本管理发布流程
- CodeForces 877C
- node.js(一)基础介绍
- 基于AE+C#实现在TOCControl中实现指定图层删除
- 美团前期运营模式(怎么吸引商家入驻?怎么吸引用户?)
- js获取当前农历时间
- linux服务器之间的文件同步(双向同步unison+inotify)
- angular.js:13920 Error: [$injector:unpr] Unknown provider: $scopeProvider - $scope - testServe