Может ли Джексон использовать конструктор со свойствами, содержащимися в java.util.Map во время (de)/ сериализации?

Вот начальная настройка, которая у меня есть

String json = "{'text': 'what is my balance', 'mid': 'D1dexnEBTCefEdRWveEt8A', 'seq': 73}";

import java.io.Serializable;
import java.time.LocalDateTime;

import org.apache.commons.lang3.builder.EqualsBuilder;

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;

@JsonDeserialize(builder = Messages.MessageBuilder.class)
public class Message implements Serializable {

  @JsonProperty("id")
  public final String id;

  @JsonProperty("botId")
  public final String botId;

  @JsonProperty("userId")
  public final String userId;

  public final String userIdKey;

  public boolean echoText=false;

  public String text;

  public String timeZone;

  public volatile String username;

  @JsonProperty("conversationId")
  public volatile String conversationId;

  public volatile int kaiUserId;

  public volatile String token;

  public final LocalDateTime inboundReceivedAt;

  public volatile LocalDateTime outboundSentAt;

  final String key;

  private final int _hashCode;

  public volatile long lastAccess = System.currentTimeMillis();

  public Message(final String pMessageId, final String pUserId, final String pBotId, final String pConversationId, final String pText) {
    id = pMessageId;
    userId = pUserId;
    botId = pBotId;
    conversationId = pConversationId;
    text = pText;
    inboundReceivedAt = LocalDateTime.now();

    key = id + "-" + userId + "-" + botId;
    userIdKey = userId + "~" + botId;

    _hashCode = key.hashCode();
  }

  void touch() { lastAccess = System.currentTimeMillis(); }

  @Override public int hashCode() { return _hashCode; }

  @Override public boolean equals(final Object pObject) {
    if (!(pObject instanceof Message)) { return false; }
    return new EqualsBuilder().append(id, ((Message)pObject).id)
                              .append(botId, ((Message)pObject).botId)
                              .append(userId, ((Message)pObject).userId)
                              .isEquals();
  }

  private static final long serialVersionUID = 42L;
}

/**
 * Jackson backed implementation of {@link Serializer}
 */
public final class JacksonSerializer implements Serializer {

    private static ObjectMapper mapper = new ObjectMapper();

    static {
        mapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);
        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
        mapper.registerModule(new JavaTimeModule());
        mapper.configure(ALLOW_SINGLE_QUOTES, true);
    }

    /**
     * For testing purposes.
     */
    static void setMapper(ObjectMapper objectMapper) {
        mapper = objectMapper;
    }

    @Override
    public <T> String serialize(T t) {
        try {
            return mapper.writeValueAsString(t);
        } catch (Exception e) {
            throw new DatabindException("Serialization error", e);
        }
    }

    @Override
    public <T> void serialize(T object, OutputStream outputStream) {
        try {
            mapper.writeValue(outputStream, object);
        } catch (Exception e) {
            throw new DatabindException("Serialization error");
        }
    }

    @Override
    public <T> void serialize(T object, File file) {
        try {
            mapper.writeValue(file, object);
        } catch (Exception e) {
            throw new DatabindException("Serialization error");
        }
    }

    @Override
    public <T> T deserialize(String s, Class<T> aClass) {
        try {
            return mapper.readValue(s, aClass);
        } catch (Exception e) {
            throw new DatabindException("Deserialization error", e);
        }
    }

    @Override
    public <T> T deserialize(InputStream inputStream, Class<T> type) {
        try {
            return mapper.readValue(inputStream, type);
        } catch (Exception e) {
            throw new DatabindException("Deserialization error", e);
        }
    }

    @Override
    public <T> T deserialize(File file, Class<T> type) {
        try {
            return mapper.readValue(file, type);
        } catch (Exception e) {
            throw new DatabindException("Deserialization error", e);
        }
    }

}

public final class Messages {


    private Messages(){}

    public static MessageBuilder standard() {
        return new MessageBuilder();
    }



    public final static class MessageBuilder {

        @JsonProperty("mid")
        String id;

        String userId;
        String botId;
        String conversationId;

        @JsonProperty("text")
        String text;



        public MessageBuilder withId(String id) {
            this.id = id;
            return this;
        }

        @JsonProperty("userId")
        public MessageBuilder withUserId(String userId) {
            this.userId = userId;
            return this;
        }

        @JsonProperty("botId")
        public MessageBuilder withBotId(String botId) {
            this.botId = botId;
            return this;
        }

        @JsonProperty("conversationId")
        public MessageBuilder withConversationId(String conversationId) {
            this.conversationId = conversationId;
            return this;
        }

        public MessageBuilder withText(String text) {
            this.text = text;
            return this;
        }

        public Message build() {
            Message result = null;
            result = new Message(id, userId, botId, conversationId, text);
            return result;
        }

    }

    public final static class PushMessageBuilder{

    }
}

@Test
public void deserializationTest() {
    Serializer serializer = new JacksonSerializer();


    String json = "{'text': 'what is my balance', 'mid': 'D1dexnEBTCefEdRWveEt8A', 'seq': 73}";
    Message message = serializer.deserialize(json, Message.class);

Я хотел бы изменить конструктор на следующую альтернативную реализацию, и в этом случае вышеупомянутая установка Джексона прекращает поиск взаимосвязи между файлом JSON и аннотациями Джексона и отображает все поля объекта Message в нули.

public final class Messages {


    private Messages(){}

    public static MessageBuilder standard() {
        return new MessageBuilder();
    }



    public final static class MessageBuilder {


       /*@JsonProperty("mid")
        String id;

        String userId;
        String botId;
        String conversationId;

        @JsonProperty("text")
        String text;*/


        @SuppressWarnings("serial")
        private Map<String, String> state = new HashMap<String, String>() {};

        @JsonProperty("mid")
        public MessageBuilder withId(String id) {
            state.put("id", id);
            //this.id = id;
            return this;
        }


        public MessageBuilder withUserId(String userId) {
            state.put("userId", userId);
            //this.userId = userId;
            return this;
        }


        public MessageBuilder withBotId(String botId) {
            state.put("botId", botId);
            //this.botId = botId;
            return this;
        }


        public MessageBuilder withConversationId(String conversationId) {
            state.put("conversationId", conversationId);
            //this.conversationId = conversationId;
            return this;
        }

        @JsonProperty("text")
        public MessageBuilder withText(String text) {
            state.put("text", text);
            //this.text = text;
            return this;
        }

        public Message build() {
            Message result = null;
            result = new Message(state.get("id"), 
                                 state.get("userId"), 
                                 state.get("botId"), 
                                 state.get("conversationId"), 
                                 state.get("text"));
            /*result = new Message(id, userId, botId, conversationId, text);*/
            return result;
        }

    }

    public final static class PushMessageBuilder{

    }
}

Это то, что поддерживает среда Джексона, и если да, что нужно изменить?

Я использую Джексон версии 2.9.4

Заранее спасибо.

1 ответ

Оказывается, эта функциональность уже присутствует в Джексоне, в то время как другая проблема затмила его для меня во время тестирования. Ниже модифицированная версия MessageBuilder демонстрирует искомую функциональность:

public final class Messages {


    private Messages(){}

    public static MessageBuilder standard() {
        return new MessageBuilder();
    }



    public final static class MessageBuilder {

        /*@JsonProperty("mid")
        String id;

        String userId;
        String botId;
        String conversationId;

        @JsonProperty("text")
        String text;*/


        @SuppressWarnings("serial")
        private Map<String, String> state = new HashMap<String, String>() {};

        @JsonProperty("mid")
        public MessageBuilder withId(String id) {
            state.put("id", id);
            //this.id = id;
            return this;
        }

        //@JsonProperty("userId")
        public MessageBuilder withUserId(String userId) {
            state.put("userId", userId);
            //this.userId = userId;
            return this;
        }

        //@JsonProperty("botId")
        public MessageBuilder withBotId(String botId) {
            state.put("botId", botId);
            //this.botId = botId;
            return this;
        }

        //@JsonProperty("conversationId")
        public MessageBuilder withConversationId(String conversationId) {
            state.put("conversationId", conversationId);
            //this.conversationId = conversationId;
            return this;
        }

        @JsonProperty("text")
        public MessageBuilder withText(String text) {
            state.put("text", text);
            //this.text = text;
            return this;
        }

        public Message build() {
            Message result = null;
            result = new Message(state.get("id"), 
                                 state.get("userId"), 
                                 state.get("botId"), 
                                 state.get("conversationId"), 
                                 state.get("text"));
            //result = new Message(id, userId, botId, conversationId, text);
            return result;
        }

    }

    public final static class PushMessageBuilder{

    }
} 

введите описание изображения здесь

Другие вопросы по тегам