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

jpa vue管理系统

jpa vue管理系统

如果您曾经在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> {
 
    @Override
    public 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);
        }
    }
 
    @Override
    public 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应用程序。

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

jpa vue管理系统

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值