私はプロジェクトで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)。ここに、私が最後に行ったデバッグ/調査をいくつか示します。
javap -v生成されたバイトコードでこれを教えてください:
MethodParameters:
Name Flags
name
maxSpeed
コンストラクターについて話すとき、-parametersフラグが実際にjavacコンパイラーに設定されていると思います。
単一のパラメーターを使用してコンストラクターを作成すると、オブジェクトが初期化されますが、複数のパラメーターコンストラクターを使用する必要があります。
各フィールドでアノテーション@JsonPropertyを使用すると同様に機能しますが、元のプロジェクトでは、コンストラクターに多くのフィールドがあるためオーバーヘッドが大きすぎます(また、アノテーションを使用してコードをリファクタリングすることは非常に困難です)。
残っている質問は:注釈なしで複数のパラメーターコンストラクターでJacksonを動作させるにはどうすればよいですか?
オブジェクトを作成するときにコンストラクターに渡す必要がある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>
簡単な答え:Java 8、javac -parameters
、および jackson-module-parameter-names
長い答え: なぜコンストラクターに@JsonCreatorの注釈が付けられている場合、その引数には@JsonPropertyの注釈が付けられているのですか?
パラメーターに@JsonProperty("xxx")
を含めると、@JsonCreator
は不要になります