Jackson+Jersey未正确序列化Date



我正在使用Jersey+Jackson开发一个REST API,但我在日期序列化方面遇到了问题。

你看,我这样配置了一个JacksonJaxbJsonProvider

@Provider
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public class JacksonProvider extends JacksonJaxbJsonProvider {
public static final ObjectMapper mapper;
static {
mapper = new ObjectMapper()
// DateUtils.ISO_DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSZ"
.setDateFormat(new SimpleDateFormat(DateUtils.ISO_DATE_FORMAT, Locale.getDefault())) 
.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
.setSerializationInclusion(JsonInclude.Include.NON_NULL);
}
public JacksonProvider() {
super();
setMapper(mapper);
}
}

然而,它正在以yyyy-MM-dd格式序列化日期,甚至没有时间信息!

在你问之前:

  • 球衣版本:2.32
  • 杰克逊版本:2.7.2
  • 是的,我知道这些版本有点旧,这是我正在尝试升级的遗留项目。当我开始的时候,它仍然使用一些SOAP端点,甚至没有使用Maven作为依赖项
  • 是的,在问之前我搜索了其他问题,我尝试了他们的解决方案,但没有成功
  • 是的,我的pom.xml配置正确
  • 是的,我的依赖项是正确的,没有一个丢失或错误,因为除了日期序列化之外,其他一切都在工作
  • 是的,我设置了一个JUnit测试,看看我的ObjectMapper是否设置正确,它的读写日期都是预期的格式
  • 是的,我知道WRITE_DATES_AS_TIMESTAMP位,不是它,我试着评论那行
  • 是的,我做了测试,以保证泽西岛实际上正在使用该提供商
  • 是的,其他ObjectMapper配置都按预期工作,只有DateFormat似乎被Jackson忽略了
  • 是的,我试过使用其他格式,但杰克逊也忽略了它们
  • 是的,我尝试了@JsonFormat注释,它很有效,但该注释是为了设置字段特定的格式,我想要的是Jackson使用的GLOBAL日期格式
  • 是的,我试过使用ContextResolver<ObjectMapper>,但它也不起作用
  • 是的,我尝试了一个快速的解决方案,用Gson代替Jackson进行测试,效果很好,但项目的所有其他部分都已经在使用Jackson了,切换会……很耗时
  • 是的,我知道有多种方法可以实现…";围绕"工作";这个问题,但我不想"围绕"工作";它,我想修复它

请记住:我的目标是为Jackson设置一个全局日期格式,以便以CLEAN、EASY的方式在序列化/反序列化中使用

欢迎任何帮助。

更新

根据要求,这里有更多的代码:

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://maven.apache.org/POM/4.0.0"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<!-- package info here -->
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.target>1.8</maven.compiler.target>
<maven.compiler.source>1.8</maven.compiler.source>
<jersey.version>2.32</jersey.version>
<jackson.version>2.7.2</jackson.version>
</properties>
<repositories>
<repository>
<id>mavenCentral</id>
<url>https://repo1.maven.org/maven2/</url>
</repository>
</repositories>
<build>
<sourceDirectory>src/java</sourceDirectory>
<testSourceDirectory>src/test/java</testSourceDirectory>
<plugins>
<plugin>
<artifactId>maven-war-plugin</artifactId>
<version>3.2.2</version>
<configuration>
<attachClasses>true</attachClasses>
<webXml>src/web/WEB-INF/web.xml</webXml>
<webResources>
<resource>
<directory>src/web</directory>
<filtering>true</filtering>
</resource>
</webResources>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.1</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
</plugins>
</build>
<dependencies>
<dependency>
<groupId>org.glassfish.jersey.core</groupId>
<artifactId>jersey-server</artifactId>
<version>${jersey.version}</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.glassfish.jersey.media/jersey-media-multipart -->
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-multipart</artifactId>
<version>${jersey.version}</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.glassfish.jersey.containers/jersey-container-servlet -->
<dependency>
<groupId>org.glassfish.jersey.containers</groupId>
<artifactId>jersey-container-servlet</artifactId>
<version>${jersey.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.jaxrs</groupId>
<artifactId>jackson-jaxrs-json-provider</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-annotations</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-core</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>42.2.18</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.0.1</version>
</dependency>
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.6</version>
</dependency>
<!--        Enable SOAP (because it still has some leftover SOAP endpoints) -->
<!-- API -->
<dependency>
<groupId>jakarta.xml.ws</groupId>
<artifactId>jakarta.xml.ws-api</artifactId>
<version>2.3.3</version>
</dependency>
<!-- Runtime -->
<dependency>
<groupId>com.sun.xml.ws</groupId>
<artifactId>jaxws-rt</artifactId>
<version>2.3.3</version>
</dependency>
</dependencies>
</project>

Application.java

@ApplicationPath("/")
public class Application extends ResourceConfig {
public Application() {
//        Register Jackson provider
register(JacksonProvider.class);
//        Register endpoint
register(ExampleEndpoint.class);
//        More registrations here...
}
}

ExampleEndpoint.java

@Path("/example")
public class ExampleEndpoint {
@GET
@Produces(MediaType.APPLICATION_JSON)
public Response getTimestamp() {
try {
Map<String, Object> response = new HashMap<>();
response.put("current_timestamp", new Date());
return Response.ok(response).build();
} catch (Exception e) {
return Response.serverError().entity(e).build();
}
}
}

由于日期格式在测试中按预期工作,但在部署应用程序时不起作用,因此需要验证部署应用程序的实际提供程序版本。

我猜(在您提供一个可复制的最小示例之前,您的类路径中有两个提供者jar,一个由应用程序服务器提供(即在Glassfish模块目录中(,另一个打包在应用程序中(或在lib/ext目录中(。这可以解释你的评论中所说的奇怪行为:

情节变得更加复杂:我实际上只是重新部署了应用程序,甚至没有更改一行代码,它就开始工作了。然后(再次没有任何更改(我再次重新部署了代码,它停止了工作。

可能是classpath顺序在部署之间发生更改,并且您的应用程序使用其他提供程序版本。

如果是这种情况,一种可行的方法是:

  1. 验证应用程序服务器提供的版本
  2. 在具有provided作用域的pom.xml中使用该确切版本(因此您将始终使用AS提供的实现

注意:这种方法将把你的应用程序绑定到AS。

最新更新