3

我有一个特定类的反序列化器,它在读取字段时需要一些排序。假设我的班级 (field1field2) 中有两个字段,为了阅读field2,它首先需要field1.

例如,对于以下 json 数据,它可以工作,因为当反序列化器解析时field2field1已经设置了:

{"field1": 3, "field2": 4}

但是,如果我们反转字段:

{"field2": 4, "field1": 3}

我需要跳过field2jp.skipChildren因为field1没有设置。当field1被解析时,杰克逊应该重新读取和解析field2.

一种选择是解析 field2 而不是跳过并将其保存在一个变量中,以便在设置 field1 时,它可以使用在 field2 中保存数据的变量。然而; 基于 的值field1,我可能不需要解析field2,因此我正在寻找更好的解决方案,因为性能在这部分代码中至关重要。

我正在使用Mapper.readValue(byte[], MyClass.class)方法,似乎杰克逊ReaderBasedJsonParser用于解析。即使可以获得令牌位置,我也找不到设置令牌位置的方法。

4

2 回答 2

3

最后我找到了一种方法。这实际上是一种解决方法,但它通过了我编写的测试。当您将字节数组传递给 mapper.readValue 时,它​​会使用ReaderBasedJsonParser它遍历数组并解析 JSON 树。

public static class SaveableReaderBasedJsonParser extends ReaderBasedJsonParser {
    private int savedInputPtr = -1;

    public SaveableReaderBasedJsonParser(IOContext ctxt, int features, Reader r, ObjectCodec codec, CharsToNameCanonicalizer st, char[] inputBuffer, int start, int end, boolean bufferRecyclable) {
        super(ctxt, features, r, codec, st, inputBuffer, start, end, bufferRecyclable);
    }

    public void save() {
        savedInputPtr = _inputPtr;
    }

    public boolean isSaved() {
        return savedInputPtr>-1;
    }

    public void load() {
        _currToken = JsonToken.START_OBJECT;
        _inputPtr = savedInputPtr;
        _parsingContext = _parsingContext.createChildObjectContext(0, 0);
    }
}

当您使用 thisJsonParser时,将传递给您的反序列化器 EventDeserializer.deserialize(JsonParser, DeserializationContext) 的JsonParser实例将是 a SaveableReaderBasedJsonParser,因此您可以安全地转换它。

当你想保存位置时,调用jp.save(),以便当你需要返回时,你可以调用只是调用jp.load()

正如我所说,这实际上是一种解决方法,但是当您需要这种功能并且出于性能原因不想两次解析树时,您可以尝试一下。

于 2015-02-15T20:52:58.957 回答
2

自定义反序列化器需要使用流 API。没有办法回去,重新解析等。

您是否为包含需要这种特殊处理的字段的字段类型或类注册了自定义反序列化器?

如果您为包含这些字段的类注册反序列化器,您可以只使用流式 API,读取实例的所有字段,将它们临时存储在例如 HashMap 中,然后分配值。

顺便说一句:你的问题闻起来像 XYProblem。也许你应该发布另一个关于原因的问题,你需要在这里解决这个问题,看看是否有更好的方法。

于 2015-02-15T07:21:15.740 回答