2012-02-26 60 views
1

我有提供商接口定製傑克遜串行器對包裝對象

interface IProvider<T> { 
    T locate(); 
} 

和包含類型IProvider的視野中的類(可以是另一種類型的其他字段)。

class MyObject { 
    MyLocator<String> field; 
} 

我需要使用Jackson 1.7將MyObject的實例序列化爲JSON。輸出必須與MyObject.field是一個字符串(即沒有引用ILocator)相同。

我不知道如何構建實現此目的所需的自定義序列化程序。這裏是我嘗試使用此任務的結構:

class MyLocatorSerializer extends SerializerBase<MyLocator<?>> { 
    public MyLocatorSerializer() { 
     super(MyLocator.class, false); 
    } 

    @Override 
    public void serialize(MyLocator<?> a_value, JsonGenerator a_jgen, 
      SerializerProvider a_provider) throws IOException, JsonGenerationException { 
     // Insert code here to serialize a_value.locate(), whatever its type 
    } 

    @Override 
    public JsonNode getSchema(SerializerProvider a_provider, Type a_typeHint) 
      throws JsonMappingException { 
     // What should I return here? I can't find documentation regarding the different schema types... 
    } 
} 

的自定義序列將使用

SimpleModule module = new SimpleModule("MyModule", new Version(1, 0, 0, null)); 
module.addSerializer(new MyLocatorSerializer()); 
objectMapper.registerModule(module); 

回答

3

另一個解決方法是在Staxman的評論之後使用混合註釋。

static class JacksonCustomModule extends SimpleModule { 
    public JacksonCustomModule() { 
     super("JacksonCustomModule", new Version(1, 0, 0, null)); 
    } 

    @Override 
    public void setupModule(SetupContext context) { 
     context.setMixInAnnotations(IProvider.class, IProviderMixIn.class); 
     super.setupModule(context); 
    } 

    interface IProviderMixIn<T> { 
     @JsonValue 
     T locate(); 
    } 
} 

激活與模塊:

objectMapper.registerModule(new JacksonCustomModule()); 
2

道歉,如果我誤解的問題進行登記,但這樣做的只是用簡單'Locate'方法上的@JsonValue,而不是編寫自定義序列化程序? 什麼@JsonValue確實是需要一個屬性的值是,並使用它,而不是創建一個JSON對象:通常這是用於序列化POJO作爲一個簡單的字符串或數字,像這樣:

public class StringWrapper { 
    @JsonValue public String value; 
} 

使類,如:

public class POJO { 
    public StringWrapper wrapped; 
} 

我們會得到系列化,如:

{ 
    "wrapper" : "string value of 'value'" 
} 

什麼,而不是原本可以看出:

{ 
    "wrapper" : { 
     "value" : "... string value ... " 
    } 
} 

註解可以明顯用於任何類型的值。

+0

謝謝您的回答。你達到的最終結果正是我所追求的。但是,就我而言,按照您的示例,StringWrapper類是第三方庫的一部分,因此使用@JsonValue對其進行註釋是不切實際的。 – bernie 2012-02-27 14:53:27

+0

像往常一樣,混合註釋[wiki.fasterxml.com/JacksonMixInAnnotations]是將這些與第三方庫一起使用的方式。 – StaxMan 2012-02-29 16:13:24

0

繼StaxMan的回答,我考察的@JsonValue運作,並得到了以下串行:

// Based on JsonValueSerializer 
private static class ProviderSerializer extends SerializerBase<IProvider<?>> { 

    public ProviderSerializer() { 
     super(IProvider.class, false); 
    } 

    @Override 
    public void serialize(IProvider<?> value, JsonGenerator jgen, SerializerProvider provider) 
      throws IOException, JsonGenerationException { 
     Object object = value.locate(); 

     // and if we got null, can also just write it directly 
     if (object == null) { 
      provider.defaultSerializeNull(jgen); 
      return; 
     } 

     Class<?> c = object.getClass(); 
     JsonSerializer<Object> ser = provider.findTypedValueSerializer(c, true, null); 
     // note: now we have bundled type serializer, so should NOT call with typed version 
     ser.serialize(object, jgen, provider); 
    } 

    @Override 
    public JsonNode getSchema(SerializerProvider provider, Type typeHint) 
      throws JsonMappingException { 
     // is this right?? 
     return JsonSchema.getDefaultSchemaNode(); 
    } 
} 

一些測試後,這確實是我所需要的。但是,我並不完全瞭解getSchema方法的用途,所以也許我做錯了什麼...