Включая существующий json через Jackson [duplicate]

Ни один из вышеперечисленных ответов не был достаточно общим для меня, поэтому я сделал это:

var someUserInputStringValue = "propertyNameOfObject i.e. 'Quantity' or 'Date'";
var SortedData = DataToBeSorted
                   .OrderBy(m => m.GetType()
                                  .GetProperties()
                                  .First(n => 
                                      n.Name == someUserInputStringValue)
                   .GetValue(m, null))
                 .ToList();

Осторожно на массивные массивы данных. Это простой код, но может вызвать у вас проблемы, если коллекция огромна, а тип объекта коллекции имеет большое количество полей. Время запуска - NxM, где:

N = # элементов в коллекции

M = # свойств в объекте

85
задан skaffman 24 January 2011 в 17:11
поделиться

10 ответов

@JsonRawValue предназначен только для сериализации, так как обратное направление немного сложнее для обработки. Фактически он был добавлен, чтобы разрешить впрыскивание предварительно закодированного контента.

Я предполагаю, что можно было бы добавить поддержку для обратного, хотя это было бы довольно неудобно: контент должен быть проанализирован, записанный обратно в «сырую» форму, которая может быть или не быть одинаковой (поскольку котировка символов может отличаться). Это для общего случая. Но, возможно, это имело бы смысл для некоторого подмножества проблем.

Но я думаю, что для вашего конкретного случая было бы полезно указать тип как «java.lang.Object», так как это должно работать нормально: для сериализации строка будет выводиться как есть, а для десериализации она десериализуется как карта. На самом деле, возможно, вы захотите иметь отдельный геттер / сеттер; getter возвращает String для сериализации (и нуждается в @JsonRawValue); и сеттер возьмет либо карту, либо объект. Вы можете перекодировать его в String, если это имеет смысл.

46
ответ дан StaxMan 19 August 2018 в 11:48
поделиться
  • 1
    Это работает как шарм; см. мой ответ для кода ( форматирование в комментариях - awgul ). – yves amsellem 12 July 2012 в 13:55
  • 2
    Для этого у меня был другой вариант использования. Похоже, если мы не хотим генерировать много строкового мусора в deser / ser, мы должны иметь возможность просто перетаскивать строку как таковую. Я видел поток, который отслеживал это, но, похоже, нет никакой возможной поддержки. Посмотрите на markmail.org/message/… – Sid 10 February 2016 в 16:09
  • 3
    @Sid нет никакого способа сделать это и токенизацию эффективно. Для поддержки сквозного прохождения необработанных токенов потребуется дополнительное сохранение состояния, что делает «регулярным» синтаксический анализ несколько менее эффективен. Это своего рода оптимизация между регулярным кодом и броском исключения: для поддержки последнего добавляются служебные данные для прежних. Джексон не был разработан, чтобы попытаться сохранить доступ к необработанным входным данным; было бы неплохо иметь его (для сообщений об ошибках тоже), но для этого потребуется другой подход. – StaxMan 12 February 2016 в 22:43

@JsonSetter может помочь. См. Мой пример («данные» должны содержать unparsed JSON):

class Purchase
{
    String data;

    @JsonProperty("signature")
    String signature;

    @JsonSetter("data")
    void setData(JsonNode data)
    {
        this.data = data.toString();
    }
}
14
ответ дан anagaf 19 August 2018 в 11:48
поделиться

У меня была аналогичная проблема, но с использованием списка с большим количеством JSON itens (List<String>).

public class Errors {
    private Integer status;
    private List<String> jsons;
}

Мне удалось сериализовать, используя аннотацию @JsonRawValue. Но для десериализации мне пришлось создать собственный десериализатор, основанный на предложении Роя.

public class Errors {

    private Integer status;

    @JsonRawValue
    @JsonDeserialize(using = JsonListPassThroughDeserialzier.class)
    private List<String> jsons;

}

Ниже вы можете увидеть мой десериализатор «Список».

public class JsonListPassThroughDeserializer extends JsonDeserializer<List<String>> {

    @Override
    public List<String> deserialize(JsonParser jp, DeserializationContext cxt) throws IOException, JsonProcessingException {
        if (jp.getCurrentToken() == JsonToken.START_ARRAY) {
            final List<String> list = new ArrayList<>();
            while (jp.nextToken() != JsonToken.END_ARRAY) {
                list.add(jp.getCodec().readTree(jp).toString());
            }
            return list;
        }
        throw cxt.instantiationException(List.class, "Expected Json list");
    }
}
1
ответ дан Biga 19 August 2018 в 11:48
поделиться

Я смог сделать это с помощью специального десериализатора (вырезать и вставить из здесь )

package etc;

import java.io.IOException;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.TreeNode;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;

/**
 * Keeps json value as json, does not try to deserialize it
 * @author roytruelove
 *
 */
public class KeepAsJsonDeserialzier extends JsonDeserializer<String> {

    @Override
    public String deserialize(JsonParser jp, DeserializationContext ctxt)
            throws IOException, JsonProcessingException {

        TreeNode tree = jp.getCodec().readTree(jp);
        return tree.toString();
    }
}
29
ответ дан Community 19 August 2018 в 11:48
поделиться
  • 1
    Я не пытался, но он должен работать: 1) сделать узел JsonNode вместо Object json 2) использовать node.asText () вместо toString (). Я не уверен в 2-м, хотя. – Vadim Kirilchuk 28 August 2015 в 16:09
  • 2
    Удивительно, насколько просто. ИМО это должно быть официальным ответом. Я пробовал с очень сложной структурой, содержащей массивы, подобъекты и т. Д. Возможно, вы отредактируете свой ответ и добавьте, что элемент String, подлежащий десериализации, должен быть аннотирован @JsonDeserialize (используя = KeepAsJsonDeserialzier.class). (и исправьте опечатку в названии вашего класса ;-) – Heri 29 September 2015 в 11:46
  • 3
    это работает для Deserializion. Как насчет сериализации raw json в pojo? Как это будет выполнено? – xtrakBandit 13 November 2015 в 01:14
  • 4
    @xtrakBandit для сериализации, используйте @JsonRawValue – smartwjw 19 January 2016 в 14:07
  • 5
    Интересно, почему getJson() действительно возвращает String. Если он только что вернул JsonNode, который был установлен через сеттер, он будет сериализован по желанию, нет? – sorrymissjackson 11 November 2016 в 08:48
  • 6
    Это работает как шарм. Спасибо Рою и @Heri. Комбинация этого сообщения вместе с комментарием Хери - это лучший ответ. – Michal 21 April 2017 в 08:33
  • 7
    @VadimKirilchuk node.asText() возвращает пустое значение напротив toString(). – v.ladynev 8 March 2018 в 12:04
  • 8
    Простое и опрятное решение. Я согласен с @Heri – mahesh 9 August 2018 в 07:18

Это даже работает в сущности JPA:

private String json;

@JsonRawValue
public String getJson() {
    return json;
}

public void setJson(final String json) {
    this.json = json;
}

@JsonProperty(value = "json")
public void setJsonRaw(JsonNode jsonNode) {
    setJson(jsonNode.toString());
}
4
ответ дан Georg 19 August 2018 в 11:48
поделиться

Вот полный рабочий пример того, как использовать модули Jackson для работы @JsonRawValue в обоих направлениях (сериализация и десериализация):

public class JsonRawValueDeserializerModule extends SimpleModule {

    public JsonRawValueDeserializerModule() {
        setDeserializerModifier(new JsonRawValueDeserializerModifier());
    }

    private static class JsonRawValueDeserializerModifier extends BeanDeserializerModifier {
        @Override
        public BeanDeserializerBuilder updateBuilder(DeserializationConfig config, BeanDescription beanDesc, BeanDeserializerBuilder builder) {
            builder.getProperties().forEachRemaining(property -> {
                if (property.getAnnotation(JsonRawValue.class) != null) {
                    builder.addOrReplaceProperty(property.withValueDeserializer(JsonRawValueDeserializer.INSTANCE), true);
                }
            });
            return builder;
        }
    }

    private static class JsonRawValueDeserializer extends JsonDeserializer<String> {
        private static final JsonDeserializer<String> INSTANCE = new JsonRawValueDeserializer();

        @Override
        public String deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException {
            return p.readValueAsTree().toString();
        }
    }
}

Затем вы можете зарегистрировать модуль после создания ObjectMapper:

ObjectMapper objectMapper = new ObjectMapper();
objectMapper.registerModule(new JsonRawValueDeserializerModule());

String json = "{\"foo\":\"one\",\"bar\":{\"A\":false}}";
Pojo deserialized = objectMapper.readValue(json, Pojo.class);
1
ответ дан Helder Pereira 19 August 2018 в 11:48
поделиться
  • 1
    Есть ли что-то еще в дополнение к вышесказанному, что вы должны делать? Я обнаружил, что метод десериализации JsonRawValue Deserialize никогда не вызывается ObjectMapper – Michael Coxon 9 August 2018 в 21:52
  • 2
    – Helder Pereira 10 August 2018 в 09:52

Это легкое решение сработало для меня:

public class MyObject {
    private Object rawJsonValue;

    public Object getRawJsonValue() {
        return rawJsonValue;
    }

    public void setRawJsonValue(Object rawJsonValue) {
        this.rawJsonValue = rawJsonValue;
    }
}

Таким образом, я смог сохранить исходное значение JSON в переменной rawJsonValue, и тогда не было никакой проблемы десериализовать его (как объект) с другими полями назад к JSON и отправить через мой REST. Использование @JsonRawValue мне не помогло, потому что сохраненный JSON был десериализован как String, а не как объект, и это было не то, что я хотел.

2
ответ дан Pavol Golias 19 August 2018 в 11:48
поделиться

Это проблема с вашими внутренними классами. Класс Pojo является [статическим внутренним классом нестатического класса вашего тестового класса, и Джексон не может создать экземпляр этого класса. Таким образом, он может сериализоваться, но не десериализоваться.

Переопределить свой класс следующим образом:

public static class Pojo {
    public String foo;

    @JsonRawValue
    public String bar;
}

Обратите внимание на добавление static

3
ответ дан skaffman 19 August 2018 в 11:48
поделиться
  • 1
    Благодарю. Это сделало меня еще на один шаг, но теперь у меня другая ошибка. Я буду обновлять исходное сообщение с новой ошибкой. – bhilstrom 24 January 2011 в 16:59

Использование объекта отлично работает в обоих направлениях ... Этот метод имеет немного избыточного десериализации исходного значения в два раза.

ObjectMapper mapper = new ObjectMapper();
RawJsonValue value = new RawJsonValue();
value.setRawValue(new RawHello(){{this.data = "universe...";}});
String json = mapper.writeValueAsString(value);
System.out.println(json);
RawJsonValue result = mapper.readValue(json, RawJsonValue.class);
json = mapper.writeValueAsString(result.getRawValue());
System.out.println(json);
RawHello hello = mapper.readValue(json, RawHello.class);
System.out.println(hello.data);

RawHello.java

public class RawHello {

    public String data;
}

RawJsonValue.java

public class RawJsonValue {

    private Object rawValue;

    public Object getRawValue() {
        return rawValue;
    }

    public void setRawValue(Object value) {
        this.rawValue = value;
    }
}
1
ответ дан Vozzie 19 August 2018 в 11:48
поделиться
46
ответ дан Community 31 October 2018 в 00:05
поделиться
Другие вопросы по тегам:

Похожие вопросы: