Jackson を使用して、ある種のダックタイピングを実装したいと思います。ここで例 6 を既に見ましたhttp://programmerbruce.blogspot.de/2011/05/deserialize-json-with-jackson-into.html。ただし、基本クラス自体が抽象クラスでない場合 (無限ループが発生する)、これは機能しません。したがって、私の質問は次のとおりです。ダックタイピングを実行して (JSON 文字列内の関連する属性を確認して)、Jackson が通常どおり逆シリアル化を行うために使用する必要がある型を返すことができる、ある種のコールバックを実装する方法はありますか?
コードの例を次に示します。サブクラスのExtendedOptionsでは機能しますが、基本クラスのOptionsでは機能しません...
public class Options {
private int size;
public int getSize() {
return size;
}
public void setSize(int size) {
this.size = size;
}
@Override
public String toString() {
return "Options";
}
}
public class ExtendedOptions extends Options {
private String feature;
public String getFeature() {
return feature;
}
public void setFeature(String feature) {
this.feature = feature;
}
@Override
public String toString() {
return "ExtendedOptions";
}
}
public class OptionsDeserializer extends
StdDeserializer<Options> {
OptionsDeserializer() {
super(Options.class);
}
@Override
public Options deserialize(JsonParser jp,
DeserializationContext ctxt) throws IOException,
JsonProcessingException {
ObjectMapper mapper = (ObjectMapper) jp.getCodec();
ObjectNode root = (ObjectNode) mapper.readTree(jp);
Iterator<Entry<String, JsonNode>> elementsIterator = root.fields();
while (elementsIterator.hasNext()) {
Entry<String, JsonNode> element = elementsIterator.next();
String name = element.getKey();
// has "feature"? => It's an ExtendedOptions object
if ("feature".equals(name)) {
return mapper.treeToValue(root, ExtendedOptions.class);
}
}
// otherwise it's just an Options object
return mapper.treeToValue(root, Options.class);
}
}
public class JacksonTest {
public static void main(String[] args) throws JsonParseException,
JsonMappingException, Exception {
String optionsString = "{ \"size\": 5 }";
String extendedOptionsString = "{ \"size\": 5, \"feature\" : \"theFeature\" }";
OptionsDeserializer deserializer = new OptionsDeserializer();
@SuppressWarnings("deprecation")
SimpleModule module = new SimpleModule(
"PolymorphicDeserializerModule", new Version(1, 0, 0,
null));
module.addDeserializer(Options.class, deserializer);
ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(module);
// works
ExtendedOptions extendedOptions = (ExtendedOptions) mapper.readValue(extendedOptionsString, Options.class);
System.out.println(extendedOptions);
// results in infinite loop!!!
Options options = mapper.readValue(optionsString, Options.class);
System.out.println(options);
}
}
編集:
StaxMan のヒントに従って BeanDeserializerModifier を実装してみました。以下は、modifyDeserializer メソッドです。
@Override
public JsonDeserializer<?> modifyDeserializer(DeserializationConfig config,
BeanDescription beanDesc, JsonDeserializer<?> deserializer) {
List<BeanPropertyDefinition> properties = beanDesc.findProperties();
for(BeanPropertyDefinition property: properties) {
String name = property.getName();
// has "feature"? => It's an ExtendedOptions object
System.out.println(name);
if("feature".equals(name)) {
System.out.println("It's an extended object!");
// should return special deserializer here...
return deserializer;
}
}
System.out.println("It's not an extended object!");
return deserializer;
}
beanDescにはOptionクラスに関する情報のみが含まれているため、これは機能しませんでした。つまり、現在の json ストリームに関する情報を取得できないため、どのデシリアライザーを返す必要があるかを判断できません。ただし、機能するソリューションを見つけましたが、100% 完璧ではありません。
@Override
public Options deserialize(JsonParser jp,
DeserializationContext ctxt) throws IOException,
JsonProcessingException {
ObjectMapper mapper = (ObjectMapper) jp.getCodec();
ObjectNode root = (ObjectNode) mapper.readTree(jp);
Iterator<Entry<String, JsonNode>> elementsIterator = root.fields();
while (elementsIterator.hasNext()) {
Entry<String, JsonNode> element = elementsIterator.next();
String name = element.getKey();
// has "feature"? => It's an ExtendedOptions object
if ("feature".equals(name)) {
return mapper.treeToValue(root, ExtendedOptions.class);
}
}
// otherwise it's just an Options object
ObjectMapper origMapper = new ObjectMapper();
return origMapper.treeToValue(root, Options.class);
}
ここでは、 ObjectMapper の新しいインスタンスを作成して、基本型Optionsを逆シリアル化します。私が言ったように、それは機能します。しかし:
1) ObjectMapper の新しいインスタンスを作成するとコストがかかる場合があります (静的属性を使用しますか?)。
2) このソリューションは、ネストされたポリモーフ オブジェクトでは機能しません。たとえば、Options に、それ自体がポリモーフである属性が含まれているとします。
別の質問があります: デシリアライザーの登録を解除する方法はありますか? もしそうなら、最後の2行を次のように置き換えることができます:
mapper.unregister(this);
Options result = mapper.treeToValue(root, Options.class);
mapper.register(this);
return result;
編集 2
わかりました、あなたは正しいです。登録解除は良い解決策ではありません。マルチスレッドについては考えていませんでした;-) とにかく、私はこれを試しました:
@Override
public JsonDeserializer<?> modifyDeserializer(DeserializationConfig config,
BeanDescription beanDesc, JsonDeserializer<?> deserializer) {
return new OptionsDeserializer();
}
しかし、それは私を同じ無限ループに導きます。