jackson+jersey未正确序列化日期

aij0ehis  于 2021-07-07  发布在  Java
关注(0)|答案(1)|浏览(333)

我正在使用jersey+jackson开发restapi,但是我在日期序列化方面遇到了问题。
你看,我配置了一个 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 Jackson版本: 2.7.2 是的,我知道那些版本有点旧,这是一个遗留项目,我正在尝试升级。当我开始的时候,它还在使用一些soap端点,甚至没有使用maven作为依赖项。。。
是的,我在问之前搜索了其他问题,我尝试了他们的解决方案,但没有结果。
是的,我的朋友 pom.xml 配置正确。
是的,我的依赖关系是正确的,没有遗漏或错误,因为其他一切都在工作,除了日期序列化。。。
是的,我设置了一个junit测试,看看我的 ObjectMapper 设置正确,读写日期均为预期格式。
是的,我知道 WRITE_DATES_AS_TIMESTAMP 一点,不是它,我试着评论那句话。
是的,我做了一些测试来保证泽西岛确实在使用那个供应商。
是的,另一个 ObjectMapper 配置按预期工作,只有 DateFormat 似乎被Jackson忽视了。
是的,我试过使用其他格式,但Jackson也忽略了它们。
是的,我试过了 @JsonFormat 注解,它是工作的,但是注解是设置字段特定的格式,我想要的是一个全局日期格式供jackson使用。
是的,我试过用 ContextResolver<ObjectMapper> 而且也不管用。
是的,我尝试了一个快速修复测试使用gson而不是jackson,它像一个魅力,但项目的所有其他部分已经使用jackson,切换将是。。。耗时。
是的,我知道有多种方法可以实现……”解决这个问题,但我不想解决它,我想解决它。
请记住:我的目标是为jackson设置一个全局日期格式,以便以干净、简单的方式用于序列化/反序列化
欢迎任何帮助。

更新

根据要求,这里还有一些代码:

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>

应用程序.java

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

示例endpoint.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();
        }
    }
}
bqujaahr

bqujaahr1#

由于日期格式在测试中按预期工作,但在部署应用程序时不起作用,因此需要验证部署应用程序时使用的实际提供程序版本。
我猜(除非您提供一个最小的可复制示例,您的类路径中有两个提供程序jar,一个由应用程序服务器提供(即glassfish modules目录),另一个打包在应用程序中(或lib/ext目录)。这可以解释你评论中所说的奇怪行为:
情节越来越复杂:我实际上只是重新部署了应用程序,甚至没有更改一行代码,它就开始工作了。然后(同样没有任何更改)我再次重新部署了代码,它停止了工作。
可能是在部署之间更改了类路径顺序,并且您的应用程序使用了其他提供程序版本。
如果是这种情况,工作方法将是:
验证应用程序服务器提供的版本
使用pom.xml中的确切版本 provided 范围(因此您将始终使用提供的实现)
注意:这种方法将把你的应用程序绑定到as。

相关问题