序列化和反序列化期间JSON属性的不同名称

kiR*_*ach 135 java json jackson

是否可能:在Jackson库中序列化/反序列化期间,在类中有一个字段,但它有不同的名称?

例如,我有"Coordiantes"课程.

class Coordinates{
  int red;
}
Run Code Online (Sandbox Code Playgroud)

对于来自JSON的反序列化,希望具有如下格式:

{
  "red":12
}
Run Code Online (Sandbox Code Playgroud)

但是当我将序列化对象时,结果应该是这样的:

{
  "r":12
}
Run Code Online (Sandbox Code Playgroud)

我尝试通过@JsonProperty在getter和setter上应用注释来实现它(具有不同的值):

class Coordiantes{
    int red;

    @JsonProperty("r")
    public byte getRed() {
      return red;
    }

    @JsonProperty("red")
    public void setRed(byte red) {
      this.red = red;
    }
}
Run Code Online (Sandbox Code Playgroud)

但我有一个例外:

org.codehaus.jackson.map.exc.UnrecognizedPropertyException:无法识别的字段"red"

bez*_*max 184

刚测试过,这个有效:

public class Coordinates {
    byte red;

    @JsonProperty("r")
    public byte getR() {
      return red;
    }

    @JsonProperty("red")
    public void setRed(byte red) {
      this.red = red;
    }
}
Run Code Online (Sandbox Code Playgroud)

这个想法是方法名称应该是不同的,因此jackson将其解析为不同的字段,而不是一个字段.

这是测试代码:

Coordinates c = new Coordinates();
c.setRed((byte) 5);

ObjectMapper mapper = new ObjectMapper();
System.out.println("Serialization: " + mapper.writeValueAsString(c));

Coordinates r = mapper.readValue("{\"red\":25}",Coordinates.class);
System.out.println("Deserialization: " + r.getR());
Run Code Online (Sandbox Code Playgroud)

结果:

Serialization: {"r":5}
Deserialization: 25
Run Code Online (Sandbox Code Playgroud)


小智 30

您可以使用在jackson 2.9.0中引入的@jsonAlias

例:

public class Info {
  @JsonAlias({ "r", "red" })
  public String r;
}
Run Code Online (Sandbox Code Playgroud)

  • [@JsonAlias的文档](http://fasterxml.github.io/jackson-annotations/javadoc/2.9.pr1/com/fasterxml/jackson/annotation/JsonAlias.html)明确指出它在序列化过程中没有任何影响其中主要名称始终使用.这不是OP想要的. (7认同)
  • @XaeroDegreaz 我猜@Asura 的意思是,你可以使用 `r` 作为主名称,但是 `red` 用于 `@JsonAlias`,它允许将其序列化为 `r`,但添加 `red` 以在反序列化时被识别. 用 `@JsonProperty("r")` 和另外的 `@JsonAlias("red")` 注释它应该可以很好地解决给定的问题。 (4认同)

DRC*_*RCB 15

我会将两个不同的getter/setter绑定到一个变量:

class Coordinates{
    int red;

    @JsonProperty("red")
    public byte getRed() {
      return red;
    }

    public void setRed(byte red) {
      this.red = red;
    }

    @JsonProperty("r")
    public byte getR() {
      return red;
    }

    public void setR(byte red) {
      this.red = red;
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 但在这种情况下,在序列化期间,我们将获得两个属性:"r"和"red",具有相同的值. (11认同)

Xae*_*eaz 12

您可以使用@JsonSetter@JsonGetter的组合来分别控制反序列化和属性的序列化.

import com.fasterxml.jackson.annotation.JsonSetter;    
import com.fasterxml.jackson.annotation.JsonGetter;

class Coordinates {
    private int red;

    //# Used during serialization
    @JsonGetter("r")
    public int getRed() {
        return red;
    }

    //# Used during deserialization
    @JsonSetter("red")
    public void setRed(int red) {
        this.red = red;
    }
}
Run Code Online (Sandbox Code Playgroud)


小智 8

@JsonAlias在 Jackson 2.9+ 中引入的注释,无需提及@JsonProperty要使用多个别名(json 属性的不同名称)反序列化的项目,效果很好。

我在我的用例com.fasterxml.jackson.annotation.JsonAlias中使用了包一致性。com.fasterxml.jackson.databind.ObjectMapper

例如:

@Data
@Builder
public class Chair {

    @JsonAlias({"woodenChair", "steelChair"})
    private String entityType;

}


@Test
public void test1() {

   String str1 = "{\"woodenChair\":\"chair made of wood\"}";
   System.out.println( mapper.readValue(str1, Chair.class));
   String str2 = "{\"steelChair\":\"chair made of steel\"}";
   System.out.println( mapper.readValue(str2, Chair.class));

}
Run Code Online (Sandbox Code Playgroud)

工作正常。


Ram*_*ich 6

可以有正常的吸气剂/设定剂对。您只需要在中指定访问模式@JsonProperty

这是为此的单元测试:

public class JsonPropertyTest {

  private static class TestJackson {

    private String color;

    @JsonProperty(value = "device_color", access = JsonProperty.Access.READ_ONLY)
    public String getColor() {
      return color;
    };

    @JsonProperty(value = "color", access = JsonProperty.Access.WRITE_ONLY)
    public void setColor(String color) {
      this.color = color;
    }

  }

  @Test
  public void shouldParseWithAccessModeSpecified() throws Exception {
    String colorJson = "{\"color\":\"red\"}";
    ObjectMapper mapper = new ObjectMapper();
    TestJackson colotObject = mapper.readValue(colorJson, TestJackson.class);

    String ser = mapper.writeValueAsString(colotObject);
    System.out.println("Serialized colotObject: " + ser);
  }
}
Run Code Online (Sandbox Code Playgroud)

我得到的输出如下:

Serialized colotObject: {"device_color":"red"}
Run Code Online (Sandbox Code Playgroud)


Вит*_*ких 6

您可以使用此变体:

import lombok.Getter;
import com.fasterxml.jackson.annotation.JsonGetter;
import com.fasterxml.jackson.annotation.JsonProperty;

//...

@JsonProperty(value = "rr") // for deserialization
@Getter(onMethod_ = {@JsonGetter(value = "r")}) // for serialization
private String rrrr;
Run Code Online (Sandbox Code Playgroud)

使用 Lombok 吸气剂


小智 5

这不是我期望的解决方案(虽然它是一个合法的用例).我的要求是允许现有的错误客户端(已经发布的移动应用程序)使用备用名称.

解决方案在于提供一个单独的setter方法,如下所示:

@JsonSetter( "r" )
public void alternateSetRed( byte red ) {
    this.red = red;
}
Run Code Online (Sandbox Code Playgroud)