web-dev-qa-db-ja.com

複数のパラメーターコンストラクターを使用したJackson JSONの逆シリアル化

私はプロジェクトでFasterXML/Jackson-Databindをしばらく使用していましたが、これを発見するまですべてがうまく機能していました post JsonPropertyアノテーション。

問題は、複数のパラメーターを取り、@ JsonCreatorアノテーションでこのコンストラクターを装飾するコンストラクターがある場合、ジャクソンは次のエラーをスローすることです。

Exception in thread "main" com.fasterxml.jackson.databind.JsonMappingException: 
Argument #0 of constructor [constructor for com.eliti.model.Cruiser, annotations: {interface com.fasterxml.jackson.annotation.JsonCreator=@com.fasterxml.jackson.annotation.JsonCreator(mode=DEFAULT)}] has no property name annotation; must have name when multiple-parameter constructor annotated as Creator
 at [Source: {
  "class" : "com.eliti.model.Cruiser",
  "inventor" : "afoaisf",
  "type" : "MeansTransport",
  "capacity" : 123,
  "maxSpeed" : 100
}; line: 1, column: 1]

問題を説明するために 小さなプロジェクト を作成しました。逆シリアル化しようとしているクラスは次のとおりです。

public class Cruise extends WaterVehicle {

 private Integer maxSpeed;

  @JsonCreator
  public Cruise(String name, Integer maxSpeed) {
    super(name);
    System.out.println("Cruise.Cruise");
    this.maxSpeed = maxSpeed;
  }

  public Integer getMaxSpeed() {
    return maxSpeed;
  }

  public void setMaxSpeed(Integer maxSpeed) {
    this.maxSpeed = maxSpeed;
  }

}

そして、逆シリアル化するコードは次のようになります。

public class Test {
  public static void main(String[] args) throws IOException {
    Cruise cruise = new Cruise("asd", 100);
    cruise.setMaxSpeed(100);
    cruise.setCapacity(123);
    cruise.setInventor("afoaisf");

    ObjectMapper mapper = new ObjectMapper().enable(SerializationFeature.INDENT_OUTPUT);
    mapper.registerModule(new ParameterNamesModule(JsonCreator.Mode.PROPERTIES));

    String cruiseJson = mapper.writeValueAsString(cruise);

    System.out.println(cruiseJson);

    System.out.println(mapper.readValue(cruiseJson, Cruise.class));

}

すでに@JsonCreatorを削除しようとしましたが、削除すると、次の例外がスローされます。

Exception in thread "main" com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of com.eliti.model.Cruise: no suitable constructor found, can not deserialize from Object value (missing default constructor or creator, or perhaps need to add/enable type information?)
 at [Source: {
  "class" : "com.eliti.model.Cruise",
  "inventor" : "afoaisf",
  "type" : "MeansTransport",
  "capacity" : 123,
  "maxSpeed" : 100
}; line: 3, column: 3]

「mvn clean install」を発行しようとしましたが、問題は解決しません。

いくつかの追加情報を含めるために、この問題について徹底的に調査しました(GitHubの問題、ブログの投稿、StackOverflowのQ&A)。ここに、私が最後に行ったデバッグ/調査をいくつか示します。

調査1

javap -v生成されたバイトコードでこれを教えてください:

 MethodParameters:
      Name                           Flags
      name
      maxSpeed

コンストラクターについて話すとき、-parametersフラグが実際にjavacコンパイラーに設定されていると思います。

調査2

単一のパラメーターを使用してコンストラクターを作成すると、オブジェクトが初期化されますが、複数のパラメーターコンストラクターを使用する必要があります。

調査3

各フィールドでアノテーション@JsonPropertyを使用すると同様に機能しますが、元のプロジェクトでは、コンストラクターに多くのフィールドがあるためオーバーヘッドが大きすぎます(また、アノテーションを使用してコードをリファクタリングすることは非常に困難です)。

残っている質問は:注釈なしで複数のパラメーターコンストラクターでJacksonを動作させるにはどうすればよいですか?

20
PedroTanaka

オブジェクトを作成するときにコンストラクターに渡す必要があるjsonプロパティの名前を指定するアノテーション@JsonPropertyを追加する必要があります。

public class Cruise extends WaterVehicle {

 private Integer maxSpeed;

  @JsonCreator
  public Cruise(@JsonProperty("name") String name, @JsonProperty("maxSpeed")Integer maxSpeed) {
    super(name);
    System.out.println("Cruise.Cruise");
    this.maxSpeed = maxSpeed;
  }

  public Integer getMaxSpeed() {
    return maxSpeed;
  }

  public void setMaxSpeed(Integer maxSpeed) {
    this.maxSpeed = maxSpeed;
  }

}

[〜#〜] edit [〜#〜]

私はちょうど以下のコードを使用してテストしました

import Java.io.IOException;

import com.fasterxml.jackson.annotation.JsonCreator.Mode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.module.paramnames.ParameterNamesModule;

class WaterVehicle {

    private String name;
    private int capacity;
    private String inventor;
    public WaterVehicle(String name) {
        this.name=name;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getCapacity() {
        return capacity;
    }
    public void setCapacity(int capacity) {
        this.capacity = capacity;
    }
    public String getInventor() {
        return inventor;
    }
    public void setInventor(String inventor) {
        this.inventor = inventor;
    }


}

 class Cruise  extends WaterVehicle{

        private Integer maxSpeed;

        public Cruise(String name, Integer maxSpeed) {
            super(name);
            this.maxSpeed = maxSpeed;
        }

        public Integer getMaxSpeed() {
            return maxSpeed;
        }

        public void setMaxSpeed(Integer maxSpeed) {
            this.maxSpeed = maxSpeed;
        }


    }

public class Test {
      public static void main(String[] args) throws IOException {
        Cruise cruise = new Cruise("asd", 100);
        cruise.setMaxSpeed(100);
        cruise.setCapacity(123);
        cruise.setInventor("afoaisf");

        ObjectMapper mapper = new ObjectMapper().enable(SerializationFeature.INDENT_OUTPUT);
        mapper.registerModule(new ParameterNamesModule(Mode.PROPERTIES));

        String jsonString = mapper.writeValueAsString( cruise);
        System.out.println(jsonString);

        Cruise anotherCruise = mapper.readValue(jsonString, Cruise.class);
         System.out.println(anotherCruise );
         jsonString = mapper.writeValueAsString( anotherCruise );
         System.out.println(jsonString);

    }

}

次の出力を生成します

{
  "name" : "asd",
  "capacity" : 123,
  "inventor" : "afoaisf",
  "maxSpeed" : 100
}
Cruise@56f4468b
{
  "name" : "asd",
  "capacity" : 123,
  "inventor" : "afoaisf",
  "maxSpeed" : 100
}

PomファイルにcompilerArgsがあることを確認してください。

<compilerArgs>
     <arg>-parameters</arg>
</compilerArgs>
21
Nithish Thomas
4
Jon Peterson

パラメーターに@JsonProperty("xxx")を含めると、@JsonCreatorは不要になります

3
sendon1982