【问题标题】:playframework: persist java8 java.time type LocalDate with hibernate in Playplayframework:在 Play 中将 java8 java.time 类型 LocalDate 保持为休眠状态
【发布时间】:2016-03-31 00:41:02
【问题描述】:

我无法为新的 java.time.* 类型进行任何类型的转换或兼容性。或者至少,LocalDate。

我看到了异常:

play.api.http.HttpErrorHandlerExceptions$$anon$1: Execution exception[[IllegalStateException: Error(s) binding form: {"dateOfBirth":["Invalid value"]}]]
    at play.api.http.HttpErrorHandlerExceptions$.throwableToUsefulException(HttpErrorHandler.scala:265) ~[play_2.11-2.4.4.jar:2.4.4]

Caused by: java.lang.IllegalStateException: Error(s) binding form: {"dateOfBirth":["Invalid value"]}
    at play.data.Form.get(Form.java:592) ~[play-java_2.11-2.4.4.jar:2.4.4]
    at controllers.Application.addPatient(Application.java:49) ~[classes/:na]

我发现了几种方法,原则上应该可以使用 JPA hibernate,但是,如果这(再次)是 Play 的问题还是什么?我不知道。

第一种方法:

提供您自己的自定义转换器:

@Converter(autoApply = true)
public class LocalDateAttributeConverter implements AttributeConverter<LocalDate, Date> 
{
    @Override
    public Date convertToDatabaseColumn(LocalDate locDate) {
        return (locDate == null ? null : Date.valueOf(locDate));
    }

    @Override
    public LocalDate convertToEntityAttribute(Date sqlDate) {
        return (sqlDate == null ? null : sqlDate.toLocalDate());
    }
}

然后在球场上需要使用这个:

@Convert(converter = LocalDateAttributeConverter.class)
public LocalDate dateOfBirth;

http://www.thoughts-on-java.org/persist-localdate-localdatetime-jpa/

当然,据我了解,原则上我不需要@Convert 注释,因为转换器本身已使用@Converter(autoApply = true) 注释进行注释。但是,由于我找不到有关在 Play 中成功使用 Converters 的文档(也没有在 google 上获得任何点击),因此我尝试过使用和不使用,但没有任何成功。

下一个方法:

嗯,实际上,据我所知,自 hibernate 5 以来,现在应该支持较新的 java8 类型......而且我已经使用了 5.0.5,并在我的类路径中包含了必要的库:

"org.hibernate" % "hibernate-java8" % "5.0.5.Final", 

https://hibernate.atlassian.net/browse/HHH-8844

这根本没有帮助。相同的堆栈跟踪。

为了更好的衡量,我添加了休眠特定的注释

@Type(type="java.time.LocalDate")

到我的领域。

那会很丑。但我会忍受它,如果它有帮助的话。它没有。同样的例外。将其与转换器一起使用会导致其他错误。这很有趣。

我正在使用 Play 2.4 和 Hibernate 5.0.5。 有没有人设法做到这一点?

【问题讨论】:

  • 我想在平安夜发布这个是个坏主意。

标签: java hibernate jpa playframework java-8


【解决方案1】:

好的,所以我终于想通了。我应该更仔细(慢慢地)查看堆栈跟踪。

其实我最后还是受到了三个问题的影响。全部以三种不同的方式解决。然而,所有这些都与对较新的 java.time 类的“待定”支持有关。公平地说,底层库的支持是最近才引入的。

第一个问题:表单绑定验证失败。

我在我的控制器中调试了表单绑定工作:

    Form<Patient> form = Form.form(Patient.class);
    form = form.bindFromRequest();
    System.out.println(form.toString());
    Patient patient = form.get();

虽然在调用 form.get() 时会导致堆栈跟踪,但实际上在调用 form.bindFromRequest() 时已经发生错误。在来自org.springframework.validation.DataBinder 的 Form.errors 映射中记录了错误。

进一步谷歌搜索,我发现了这个:

https://groups.google.com/forum/#!topic/play-framework/Wl_ip56111c

实施解决了我的第一个问题,然后表单绑定起作用了。

第二个问题:持久性

我正在使用 Hibernate 来持久化对象。 将 hibernate-java8 添加到我的类路径中可以持久化新数据类型。对于第 2 场比赛,这在您的 build.sbt 文件中。

libraryDependencies ++= Seq(
  javaJdbc,
  cache,
  javaWs,
  javaJpa,
  "org.hibernate" % "hibernate-entitymanager" % "5.0.5.Final",
  "org.hibernate" % "hibernate-java8" % "5.0.5.Final", 
)

https://hibernate.atlassian.net/browse/HHH-8844

第三个问题:Json 支持

我无法将我的模型实例化呈现为 Json 对象。应该支持它,但不知何故,在游戏中,它没有设置。

我发现这篇文章向我展示了如何解决这个问题:

    // http://stackoverflow.com/questions/32872474/how-to-use-java-time-localdate-on-a-play-framework-json-rest/32891177#comment56814598_32891177
    // https://groups.google.com/forum/?hl=en.#!searchin/play-framework/java.time.localdate/play-framework/Dv-IpvBqWgo/l6NTp3e0BQAJ

为了您的方便,我在这里发布的修复:

    ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new Jdk8Module());
    mapper.registerModule(new JSR310Module());
    mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    Json.setObjectMapper(mapper);

第一个和第三个修复需要在应用程序启动时执行。对于 play 2.4,GlobalSettings 对象已被弃用,所以我不确定他们希望你如何进行这种初始化(有人?)。但是,幸运的是它仍然可以工作。我当然没有 Globals 类(不再默认创建),所以我创建了该类并使用以下键将我的应用程序指向它(来自 application.conf 文件):

# Minimal global settings to fix form binding and json support of java.time.LocalDate
application.global=GlobalFixes

为了您的方便,我将整个课程发布在这里,供那些仍在学习 Playframework 的人使用

import play.*;
import play.data.format.Formatters;
import play.libs.Json;

import java.time.LocalDate;
import java.util.Locale;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;
import com.fasterxml.jackson.datatype.jsr310.JSR310Module;

import java.text.ParseException;

/**
 * Fixes for play application. TODO remove this once Play has fixed support for LocalDate
 * @author Sean van Buggenum
 *
 */
public class GlobalFixes extends GlobalSettings
{   // One can see from the javadoc of java.time.LocalDate that the toString guarantees this format, regardless of locale!
    private static final Pattern datePattern = Pattern.compile("(\\d\\d\\d\\d)-(\\d\\d)-(\\d\\d)");

    public void onStart(Application app)
    {
        //https://groups.google.com/forum/#!topic/play-framework/Wl_ip56111c
        Formatters.register(LocalDate.class, new Formatters.SimpleFormatter<LocalDate>()
        {
            @Override
            public LocalDate parse(String input, Locale l) throws ParseException
            {
                Matcher m = datePattern.matcher(input);
                if (!m.matches())
                    throw new ParseException("No valid Input for date text: " + input, 0);
                return LocalDate.of(Integer.valueOf(m.group(1)), Integer.valueOf(m.group(2)), Integer.valueOf(m.group(3)));
            }

            @Override
            public String print(LocalDate localTime, Locale l)
            {
                return localTime.toString();
            }
        });

        // Add Json's java 8 support
        // http://stackoverflow.com/questions/32872474/how-to-use-java-time-localdate-on-a-play-framework-json-rest/32891177#comment56814598_32891177
        // https://groups.google.com/forum/?hl=en.#!searchin/play-framework/java.time.localdate/play-framework/Dv-IpvBqWgo/l6NTp3e0BQAJ
        ObjectMapper mapper = new ObjectMapper();
        mapper.registerModule(new Jdk8Module());
        mapper.registerModule(new JSR310Module());
        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        Json.setObjectMapper(mapper);
    }
}

我希望这对某些人有所帮助。

【讨论】:

  • 注意,上述问题和答案是关于 Play v 2.4 在测试 Play 2.6 及其包含的 Json 支持时,不需要额外的修复。关于 Hibernate,不需要包含任何“附加”库,如 hibernate-java8。但是,仍然需要创建一个自定义格式化程序来处理表单绑定中的 LocalDate。
猜你喜欢
  • 1970-01-01
  • 2012-11-12
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-08-04
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多