ObjectMapper can't deserialize without default constructor after upgrade to Spring Boot 2

前端 未结 7 1595
予麋鹿
予麋鹿 2020-12-24 01:42

I have following DTOs:

@Value
public class PracticeResults {
    @NotNull
    Map wordAnswers;
}

@Value
public class ProfileMetaDto {

         


        
相关标签:
7条回答
  • 2020-12-24 01:42

    Due to breaking changes in Lombok version 1.16.20 you need to set the following property in your lombok.config file (if you don't have this file you can create it in your project root):

    lombok.anyConstructor.addConstructorProperties=true
    

    This is described in the Lombok changelog: https://projectlombok.org/changelog.

    After that the @Value should be accepted again by Jackson.

    You may be interested in following the related GitHub issue here, although it's about @Data: https://github.com/rzwitserloot/lombok/issues/1563

    0 讨论(0)
  • 2020-12-24 01:44

    In case you have some final fields within your class and you would rather have Object creator to use specific constructor to deserialize your POJO along with @Data annotation, I suggest annotating the constructor with @JsonCreator(mode = JsonCreator.Mode.PROPERTIES) and concrete required constructor parameters with @JsonProperty("<field_name>"). The example can be seen below:

    package test.model;
    
    import com.fasterxml.jackson.annotation.JsonCreator;
    import com.fasterxml.jackson.annotation.JsonProperty;
    import lombok.Data;
    
    import java.util.Date;
    
    @Data
    public class KafkaEventPojo {
        private final String executionId;
        private final String folder;
        private final Date created_at;
    
        private Date updated_at;
        // ... other params
        
        /**
         * Instantiates a new Kafka event pojo.
         *
         * @param executionId the execution or flow id of the event lifecycle
         * @param folder      the folder to be retrieved
         */
        @JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
        public KafkaEventPojo(@JsonProperty("executionId") String executionId, @JsonProperty("folder") String folder) {
            this(executionId, folder, new Date(System.currentTimeMillis()));
        }
    
        private KafkaEventPojo(String executionId, String folder, Date date) {
            this.executionId = executionId;
            this.folder = folder;
    
            this.created_at = date;
            this.updated_at = date;
        }
        
        // ... other logic
    }
    
    0 讨论(0)
  • 2020-12-24 01:46

    I have upgraded lombok version to : 'org.projectlombok:lombok:1.18.0' and it worked for me.

    0 讨论(0)
  • 2020-12-24 01:51

    One more way to solve this problem. Use Jackson parameter names module, which is included in spring boot 2 by default. After this Jackson can deserialize objects. But it works only if you have more than 1 property in object. In case of single property I receive following error message:

    com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance of `SomeClassName` (no Creators, like default construct, exist): cannot deserialize from Object value (no delegate- or property-based Creator)
    

    Because of the following:

    Marker annotation that can be used to define constructors and factory methods as one to use for instantiating new instances of the associated class.

    NOTE: when annotating creator methods (constructors, factory methods), method must either be:

    • Single-argument constructor/factory method without JsonProperty annotation for the argument: if so, this is so-called "delegate creator", in which case Jackson first binds JSON into type of the argument, and then calls creator. This is often used in conjunction with JsonValue (used for serialization).
    • Constructor/factory method where every argument is annotated with either JsonProperty or JacksonInject, to indicate name of property to bind to

    Also note that all JsonProperty annotations must specify actual name (NOT empty String for "default") unless you use one of extension modules that can detect parameter name; this because default JDK versions before 8 have not been able to store and/or retrieve parameter names from bytecode. But with JDK 8 (or using helper libraries such as Paranamer, or other JVM languages like Scala or Kotlin), specifying name is optional.

    To handle this case with Lombok I've used following workaround:

    @Value
    @AllArgsConstructor(onConstructor = @__(@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)))
    class SomeClassName {...}
    
    0 讨论(0)
  • 2020-12-24 02:00

    As others have said you need to add more the the @Value annotation to tell Lombok to write the @JsonCreator on the generated constructor

    @AllArgsConstructor(onConstructor_={@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)})
    

    Note a slightly different syntax from other answers here if you are using JDK8+ (this is mentioned in the javadoc)

    If you are using the freefair gradle plugin you also need to configure it to add constructor properties in build.gradle:

    lombok {
        config['lombok.anyConstructor.addConstructorProperties'] = 'true'
    }
    
    0 讨论(0)
  • 2020-12-24 02:03

    I had this issue and the solution that worked for me what creating a default constructor without fields and the problem disappeared.

    0 讨论(0)
提交回复
热议问题