在泽西岛使用Jackson配置多个ObjectMappers

cma*_*son 10 jax-rs jersey jackson jackson-modules

是否可以使用Jackson设置Jersey以使用多个配置进行序列化/反序列化ObjectMappers

我希望能够做的是注册一个"默认"杰克逊ObjectMapper,然后能够注册另一个功能,提供ObjectMapper一些特殊配置,在某些情况下将"覆盖""默认" ObjectMapper.

例如,这ContextResolver将是"默认"映射器:

@Provider
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public class JacksonMapperProvider implements ContextResolver<ObjectMapper> {
    private final ObjectMapper mObjectMapper;

    public JacksonMapperProvider() {
        mObjectMapper = createMapper();
    }

    protected abstract ObjectMapper createMapper() {
        ObjectMapper mapper = createMapper();

        return mapper
            .setSerializationInclusion(Include.ALWAYS)
            .configure(JsonParser.Feature.ALLOW_COMMENTS, true)
            .configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true)
            .configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true)
            .configure(JsonParser.Feature.ALLOW_UNQUOTED_CONTROL_CHARS, true);
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        return mObjectMapper;
    }
}
Run Code Online (Sandbox Code Playgroud)

ContextResolver将覆盖"默认"映射器:

@Provider
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public class SpecializedMapperProvider implements ContextResolver<ObjectMapper> {
    private final ObjectMapper mObjectMapper;

    public SpecializedMapperProvider() {
        mObjectMapper = createMapper();
    }

    protected abstract ObjectMapper createMapper() {
        ObjectMapper mapper = createMapper();

        return mapper
            .disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
            .setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS"))
            .registerModule(new SpecializedModule1())
            .registerModule(new SpecializedModule2());
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        if(SomeType.isAssignableFrom(type)) {
            return mObjectMapper;
        }
        return null;
    }
}
Run Code Online (Sandbox Code Playgroud)

我在JacksonJsonProvider代码中看到Jackson支持ObjectMapper提供者注入/解析.然而,在实践中,我所看到的是提供者的"顺序"似乎是随机的(我猜它不是,但我不能理清如何控制顺序).有时"覆盖"出现在"默认"之前,一切正常,但在下次服务器启动时,订单会发生变化.

我试图通过多种方式实现这一目标,包括:

  • ContextResolver<ObjectMapper>手动注册实现(以不同的顺序)
  • ContextResolver<ObjectMapper>通过@Provider注释注册实现
  • 注册时指定优先级

我使用以下内容:

  • 泽西岛2.8
  • 杰克逊2.3.3

也许我采取了一种完全不正确的方法?
有没有更好的方法来实现我想要做的事情?也许我应该只定义两个独立的JAX-RS应用程序,并为每个应用程序配置一个ObjectMapper?

Ald*_*den 4

您可以配置提供程序的顺序,但实际上在这种情况下最好使用一个提供程序:

@Provider
public class JacksonMapperProvider implements ContextResolver<ObjectMapper> {
    private final ObjectMapper defaultMapper;
    private final ObjectMapper specializedMapper;

    public JacksonMapperProvider() {
        defaultMapper = createDefaultMapper();
        specializedMapper = createSpecializedMapper();
    }

    private static ObjectMapper createDefaultMapper() {
        return new ObjectMapper()
            .setSerializationInclusion(Include.ALWAYS)
            .configure(JsonParser.Feature.ALLOW_COMMENTS, true)
            .configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true)
            .configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true)
            .configure(JsonParser.Feature.ALLOW_UNQUOTED_CONTROL_CHARS, true);
    }

    private static ObjectMapper createSpecializedMapper() {
        return new ObjectMapper()
            .disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
            .setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS"))
            .registerModule(new SpecializedModule1())
            .registerModule(new SpecializedModule2());
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        if (SomeType.isAssignableFrom(type)) {
            return specializedMapper;
        }
        else {
            return defaultMapper;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 谢谢。这是我最初为了让它工作而做的,但是,我希望将应用程序的不同模块/区域分开并且不可知,并避免可能因需要不同的映射器而导致潜在的“大型提供者”。您能否详细说明为什么_“在这种情况下最好使用一个提供商”_? (2认同)