2
如何自定义Jackson反序列化一个包含${token}
的值?Jackson可变插值反序列化
这里是我想从Apache Commons Configuration variable interpolation添加functionnality的例子:
{
"file" = "${sys:user.home}/path/to/my_file"
}
如何自定义Jackson反序列化一个包含${token}
的值?Jackson可变插值反序列化
这里是我想从Apache Commons Configuration variable interpolation添加functionnality的例子:
{
"file" = "${sys:user.home}/path/to/my_file"
}
您可以注册custom string deserialiser基于这将在deserialisation后插值变量的默认。
但是,正如在评论中指出的那样,这对于诸如文件和URL之类的非字符串类型不起作用。更好的想法是通过传递定制的JsonFactory
来覆盖JsonParser
的getText()
和getValueAsString()
方法。
下面是一个例子:
public class JacksonInterpolateString {
static final String JSON = "{ \"file\":\"${sys:user.home}/path/to/my_file\" }";
public static class Bean {
public File file;
@Override
public String toString() {
return file.toString();
}
}
private static class MyJsonParser extends JsonParserDelegate {
public MyJsonParser(final JsonParser d) {
super(d);
}
@Override
public String getText() throws IOException {
final String value = super.getText();
if (value != null) {
return interpolateString(value);
}
return value;
}
@Override
public String getValueAsString() throws IOException {
return getValueAsString(null);
}
@Override
public String getValueAsString(final String defaultValue) throws IOException {
final String value = super.getValueAsString(defaultValue);
if (value != null) {
return interpolateString(value);
}
return null;
}
}
private static class MyMappingJsonFactory extends MappingJsonFactory {
@Override
protected JsonParser _createParser(
final char[] data,
final int offset,
final int len,
final IOContext ctxt,
final boolean recyclable)
throws IOException {
return new MyJsonParser(super._createParser(data, offset, len, ctxt, recyclable));
}
@Override
protected JsonParser _createParser(final Reader r, final IOContext ctxt)
throws IOException {
return new MyJsonParser(super._createParser(r, ctxt));
}
}
private static String interpolateString(final String value) {
return value.replace("${sys:user.home}", "/home/user");
}
public static void main(String[] args) throws IOException {
final JsonFactory factory = new MyMappingJsonFactory();
final ObjectMapper mapper = new ObjectMapper(factory);
System.out.println(mapper.readValue(JSON, Map.class));
System.out.println(mapper.readValue(JSON, Bean.class));
}
}
输出:
{file=/home/user/path/to/my_file}
/home/user/path/to/my_file
如果我想反序列化'String',但如何应用在每个解串器型的转变他的解决方案仅适用(即路径)?我想我必须创建一个'JsonParser'来覆盖'getText'或'getValusAsString'? – 2014-10-29 10:41:14
@gulecroc好点!我已经用你的建议更新了我的答案。 – 2014-10-29 11:51:08
听起来不错!我被重写了'JsonParser',不知道'JsonParserDelegate'!抱歉没有足够的声望投票了...所以+1 – 2014-10-29 13:08:23