如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱。 使用enum作为@Entity的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它们。 它给了您2 + 1个选择:

托肖夫达林

  1. @Enumerated(EnumType.ORDINAL) (默认值)将使用Enum.ordinal()映射enum值。 基本上,第一个枚举值将在数据库列中映射为0 ,第二个映射为1 ,依此类推。这非常紧凑,并且在要修改枚举时非常有用。 在中间删除或增加值或重新排列它们将完全破坏现有记录。 哎哟! 更糟糕的是,单元测试和集成测试通常在干净的数据库上运行,因此它们不会发现旧数据之间的差异。
  2. @Enumerated(EnumType.STRING)更安全,因为它存储enum字符串表示形式。 现在,您可以安全地添加新值并移动它们。 但是,重命名Java代码中的enum仍会破坏DB中的现有记录。 更重要的是,这种表示非常冗长,不必要地消耗了数据库资源。
  3. 您还可以使用原始表示形式(例如single charint ),并在@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;//...
}

其中Coverenum

public enum Cover {PAPERBACK, HARDCOVER, DUST_JACKET}

ORDINALSTRING都不是一个很好的选择。 前者是因为以任何方式重新排列前三个值都会破坏现有记录的加载。 后者太冗长。 这是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.BigDecimalscala.Option或小写类。 在Java中,最终将有一种可移植的方式来映射Joda时间 。 最后但并非最不重要的一点是,如果您喜欢(非常)强类型的域,则可能希望拥有PhoneNumber类(带有isInternational()getCountryCode()和自定义验证逻辑),而不是Stringlong 。 JPA 2.1中的这一小增加肯定会显着提高域对象的质量。

如果您想使用此功能, 可以在GitHub上找到示例Spring Web应用程序。

参考: Java和社区博客中的JCG合作伙伴 Tomasz Nurkiewicz 在JPA 2.1中使用@Convert进行了正确的映射枚举 。

翻译自: https://www.javacodegeeks.com/2013/06/mapping-enums-done-right-with-convert-in-jpa-2-1.html

在JPA 2.1中使用@Convert正确完成映射枚举相关推荐

  1. jpa vue管理系统_在JPA 2.1中使用@Convert正确完成映射枚举

    jpa vue管理系统 如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱. 使用enum作为@Entity的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它 ...

  2. 1.0jpa 2.0_在JPA 2.1中使用@Convert正确完成映射枚举

    1.0jpa 2.0 如果您曾经在JPA中使用过Java枚举,那么您肯定会意识到它们的局限性和陷阱. 使用enum作为@Entity的属性通常是一个很好的选择,但是2.1之前的JPA不能很好地处理它们 ...

  3. JPA 2.0 中的动态类型安全查询

    如何通过 Criteria API 构建动态查询并减少运行时失败 如果编译器能够对查询执行语法正确性检查,那么对于 Java 对象而言该查询就是类型安全的.Java™Persistence API ( ...

  4. JPA 2.0 中的动态类型安全查询(如何通过 Criteria API 构建动态查询并减少运行时失败)...

    简介:  如果编译器能够对查询执行语法正确性检查,那么对于 Java 对象而言该查询就是类型安全的.Java™Persistence API (JPA) 的 2.0 版本引入了 Criteria AP ...

  5. 学习网页制作中如何在正确选取和使用 CSS 单位

    在 CSS 测量系统中,有好几种单位,如像素.百分比.英寸.厘米等等,Web 开发人员很难了解哪些单位在何处使用,如何使用.很多人习惯了总是使用同一种单位,但这一决定可能会严重限制你的设计的执行. 这 ...

  6. 使用JPA获取Oracle中的日期字段丢失时分秒

    2019独角兽企业重金招聘Python工程师标准>>> 使用JPA获取oracle中的日期字段发现没有时分秒,因为之前没有关注过这个字段, 最近由于需要,却发现时分秒奇怪的丢失.   ...

  7. sqlserver中,如果正确得到中文的长度

    sqlserver中,如果正确得到中文的长度:             len('wangzhq')=7          len('代码')=2             正确:       data ...

  8. 如何显示服务器上excel,如何在HTML中使用OWC正确显示、加载服务器端的excel文件呢?...

    如何在HTML中使用OWC正确显示.加载服务器端的excel文件呢? 我的服务器端是unix操作系统,应用是java写的 我试了很多方法,都无法将excel在浏览器中以只读的形式显示出来,环境是XP  ...

  9. 下列关于python的说法中、正确的是-以下关于 Python版本的说法中,哪个是正确的?...

    [单选题]以下赋值语句中合法的是 [单选题]字符串:s= abcde',n=len(s).索引字符串s字符'c',哪个语句是正确的? [单选题]以下哪个方法能够根据','分隔字符串? [单选题]len ...

最新文章

  1. [Zend PHP5 Cerification] Lectures -- 4. XML Web Service
  2. js插件---10个免费开源的JS音乐播放器插件
  3. 我国java发展_Java在我国的应用现状和发展趋势
  4. CoreNLP请求超时 runtime out
  5. pythonandroid自动化测试仪器_使用Python进行Android自动化测试
  6. 技术动态 | ACL 2019 知识图谱的全方位总结
  7. 飞鸽传书 再次报道,回到90年代中期的业界
  8. SSRF libcurl protocol wrappers利用分析
  9. SQLyog下载与安装
  10. SQL Server不允许保存更改的解决方法
  11. 安装Anaconda之后cmd打不开
  12. logback配置文件---logback.xml详解
  13. 一文读懂上拉电阻:工作原理和阻值确定
  14. (一) Vue在创建的时候 入口文件 及相关的路由配置(及子路由配置)
  15. 紫外线测试仪软件,紫外荧光老化测试仪
  16. pgsql 比较数字字符串_postgreSQL数据类型转换字符串和数值
  17. idea使用svn报错-Error:Can not get current revision for file D:/IDEADire...
  18. Office 2007 使用心得
  19. 自定义view-滑动刻度尺计算金额
  20. 技能在赛题解析:交换机防环路设置

热门文章

  1. java数据库编程——元数据(metadata)+web 与企业应用中的连接管理
  2. Ubuntu下MySQL、Redis以及MongoDB三个数据库的启动、重启以及停止命令
  3. javafx 图标_JavaFX技巧32:需要图标吗? 使用Ikonli!
  4. camel apache_Apache Camel 3只有2个月的路程
  5. aws lambda使用_使用AWS Lambda,S3和AWS CloudFront进行动态内容缓存
  6. 安装openstack_午餐前如何安装OpenStack Cloud
  7. java lambda循环_使用Java 8 Lambda简化嵌套循环
  8. junit4 集成测试_使用JUnit规则进行干净的集成测试
  9. jboss4 迁移_应用程序服务器迁移:从JBoss EE5到Wildfly EE7
  10. sts集成jboss_与JBoss BPM Travel Agency更新了Modern BPM数据集成