3

像往常一样,我的问题可能有一个非常简单的解决方案:

我有一个JSON 模式片段,它定义了以下枚举:

"title" : {
    "type": "string",
    "enum": ["Mr", "Miss", "Mrs", "Ms"],
    "description": "The person's title"
 }

然后,我公司的框架使用jsonschema2pojo和 maven 来创建必要的 POJO(Title位于Clazz,作为JSON 模式title的一部分clazz- 以 clazz 作为名称 - 将其替换为员工或客户或任何你喜欢的东西):

生成的 POJO

@Generated("org.jsonschema2pojo")
public static enum Title {

    MR("Mr"),
    MISS("Miss"),
    MRS("Mrs"),
    MS("Ms");
    private final String value;
    private static Map<String, Clazz.Title> constants = new HashMap<String, Clazz.Title>();

    static {
        for (Clazz.Title c: values()) {
            constants.put(c.value, c);
        }
    }

    private Title(String value) {
        this.value = value;
    }

    @JsonValue
    @Override
    public String toString() {
        return this.value;
    }

    @JsonCreator
    public static Clazz.Title fromValue(String value) {
        Clazz.Title constant = constants.get(value);
        if (constant == null) {
            throw new IllegalArgumentException(value);
        } else {
            return constant;
        }
    }
}

当我针对它运行包含以下内容的请求时:

...
  "title" : "Mr",
...

我收到了这个错误

com.fasterxml.jackson.databind.exc.InvalidFormatException: Can not construct instance of com.example.foo.representations.jaxb.Clazz$Title from String value 'Mr': value not one of declared Enum instance names: [Mr, Miss, Mrs, Ms]  at [Source: org.apache.cxf.transport.http.AbstractHTTPDestination$1@1a372b7; line: 4, column: 3] (through reference chain: com.example.foo.representations.jaxb.MySchema["Clazz"]->com.example.foo.representations.jaxb.Clazz["title"])

显然,“先生”在枚举中。

调试时,我可以看到它通过以下类(堆栈)运行:

findEnum():120, EnumResolver (com.fasterxml.jackson.databind.util)
deserialize():79, EnumDeserializer (com.fasterxml.jackson.databind.deser.std)

看起来他们只对枚举的“键”(即常量,例如“ MR”而不是“ Mr”)感兴趣。我猜@JsonCreator注释由于某种原因被忽略了。

知道如何解决这个问题吗? 是否有可能在任何地方设置可能导致此行为的配置值?(我正在做一个大项目;如果我知道我需要寻找什么,我可以搜索代码库;也许另一个开发人员在某处“错误配置”了某些东西......)或者问题可能出TitleClazz哪里?我需要投入大量@JsonProperty资金吗?(如果是这样,具体如何?)

我们正在使用 jackson-core、-annotations 和 -databind 2.4.2。

更新:我将其作为一个独立项目进行了尝试,使用以下代码,并且它完美地工作 - 这意味着必须有某种设置可以防止注释被考虑......

ObjectMapper mapper = new ObjectMapper(); // create once, reuse
Clazz value = mapper.readValue(new File("resources/data.json"), Clazz.class);
4

1 回答 1

3

因此,事实证明我们在 Spring 配置中有以下内容:

<bean id="mapper" class="com.example.foo.jaxb.links.JsonMapper" />
<!-- ... -->
       <jaxrs:providers>
           <bean id="jsonprovider"
                 class="com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider">
               <constructor-arg ref="mapper" />
               <constructor-arg>
                   <value></value>
               </constructor-arg>
           </bean>
       </jaxrs:providers>

Jsonmapper 扩展com.fasterxml.jackson.databind.ObjectMapper,并且主要在其构造函数中自行设置 Jackson 设置。然而,我在绝望中错过的一件事是这段代码:

// use JAXB annotations (only) 
setAnnotationIntrospector(new JaxbAnnotationIntrospector(
    com.fasterxml.jackson.databind.type.TypeFactory.defaultInstance()));

看起来,它几乎做到了它在锡上所说的:它阻止了杰克逊评估@JsonCreator@JsonValue注释。一旦“杰克逊注释抑制器”消失,一切都很好。

我想做的是了解它实际上是如何工作的,所以如果有人对 Spring 的东西或注释的抑制有任何有用的链接到文档/how-tos/manuals/books,那将不胜感激。但与此同时,这解决了我的问题。:)

于 2015-07-24T13:49:16.190 回答