Gson自定义类型适配器仅在一个对象类型上序列化null?

Lou*_*rda 3 java serialization json gson

我想在我的JSON主体中仅对一个PUT上的一个类型序列化null.我不想为对象中的任何其他类型序列化null.我有这样的事情:

public class Patient {
    public Address address;
    public String first_name;
    public String last_name;
}
Run Code Online (Sandbox Code Playgroud)

如果地址为空,我只想序列化地址.所以举个例子

Patient patient = new Patient();
patient.address = null;
patient.last_name = "Doe";
Run Code Online (Sandbox Code Playgroud)

看起来像这样:

"address":null,
"last_name":"Doe"
Run Code Online (Sandbox Code Playgroud)

如果将地址指定为null,则患者不会离开对象,因为默认情况下Gson不会序列化我想要的空值,并且姓氏保留指定的字符串值.

我可以使用Gson自定义类型适配器吗?

public class GsonCustomAdapter extends TypeAdapter<Address>
Run Code Online (Sandbox Code Playgroud)

我对这个概念并不熟悉,并且一直试图理解它.任何帮助深表感谢.

Ale*_* C. 9

如果默认情况下您不想序列化空值,JsonWriter则只有在实际读取Address实例时才能告诉它序列化它.

我们假设以下课程:

class Address {
    public String country = "UK";
    public String city = "London";
}
Run Code Online (Sandbox Code Playgroud)

现在我们为Address类创建一个特定的类型适配器.在这里你明确地说,即使JsonWriter不应该null在响应中写入值,你只允许它为Address字段这样做(参见代码中的注释).

class AddressAdapter extends TypeAdapter<Address> {
    @Override
    public void write(JsonWriter out, Address address) throws IOException {
        if (address == null) {
            //if the writer was not allowed to write null values
            //do it only for this field
            if (!out.getSerializeNulls()) {
                out.setSerializeNulls(true);
                out.nullValue();
                out.setSerializeNulls(false);
            } else {
                out.nullValue();
            }
        } else {
            out.beginObject();
            out.name("country");
            out.value(address.country);
            out.name("city");
            out.value(address.city);
            out.endObject();
        }
    }

    @Override
    public Address read(JsonReader in) throws IOException {
        if(in.peek() == JsonToken.NULL) {
            in.nextNull();
            return null;
        }
        in.beginObject();
        Address address = new Address();
        in.nextName();
        address.country = in.nextString();
        in.nextName();
        address.city = in.nextString();
        in.endObject();
        return address;
    }
}
Run Code Online (Sandbox Code Playgroud)

现在您必须注册此适配器,以便解析器知道在序列化/反序列化Address字段时必须使用它.为此,请使用注释@JsonAdapter.

class Patient {
    @JsonAdapter(AddressAdapter.class)
    public Address address;
    public String first_name;
    public String last_name;
}
Run Code Online (Sandbox Code Playgroud)

它已经完成了!

例如,让我们以你的例子为例:

Patient patient = new Patient();
patient.last_name = "Doe";
Run Code Online (Sandbox Code Playgroud)

将解析器设置为序列化空值后,您将获得:

{"address":null,"first_name":null,"last_name":"Doe"}
Run Code Online (Sandbox Code Playgroud)

当您不允许它时(默认设置):

{"address":null,"last_name":"Doe"}
Run Code Online (Sandbox Code Playgroud)

通过为患者设置地址:

patient.address = new Address();
...
{"address":{"country":"UK","city":"London"},"last_name":"Doe"}
Run Code Online (Sandbox Code Playgroud)

作为一个注释,如果你想在Java端坚持命名约定,你可以使用注释@SerializedName,例如:

@SerializedName("first_name") public String firstName;
@SerializedName("last_name")  public String lastName;
Run Code Online (Sandbox Code Playgroud)

希望能帮助到你 !:-)