gson 使用两个具有相同SerializedName的字段类

1l5u6lss  于 2022-11-06  发布在  其他
关注(0)|答案(2)|浏览(154)

我的服务遇到了一个问题,因为同一个字段在请求和响应中的形式不同,因此我使用@Expose注解来序列化一个字段并反序列化另一个字段:

@SerializedName("photo")
@Expose(deserialize = false)
private String imageB64;

@SerializedName("photo")
@Expose(serialize = false)
private ImageURL imageURL;

但在响应中,我的服务却发起了一个异常:

Caused by: java.lang.IllegalArgumentException: class User declares multiple JSON fields named photo

我正在使用带有GSON转换器的Retrofit。
谢谢你的帮助!

nwsw7zdq

nwsw7zdq1#

我认为不可能添加多个@SerializedName注解,因为否则您将得到您提供的错误。
但是,您可以创建一个自定义的TypeAdapter来手动处理对象的序列化/反序列化,如下所示:

我的对象

public class MyObject {

  private String url;

  private int number;

  // constructor + getters + setters + custom "toString()"
  ...

}

我的对象类型适配器

class MyObjectTypeAdapter extends TypeAdapter<MyObject> {

  @Override
  public void write(JsonWriter out, MyObject value) throws IOException {
    out.beginObject().name("photo").value(value.getUrl()).endObject();
  }

  @Override
  public MyObject read(JsonReader in) throws IOException {
    MyObject result = new MyObject();
    in.beginObject();
    while (in.hasNext()) {
      switch (in.nextName()) {
      case "photo":
        result.setNumber(in.nextInt());
      }
      // other fields
      ...
    }

    in.endObject();
    return result;
  }
}

您可以按以下方式使用它:

public static void main(String[] args) {
  Gson gson = new GsonBuilder().registerTypeAdapter(MyObject.class, new MyObjectTypeAdapter()).create();

  System.out.println(gson.toJson(new MyObject("myUrl", 1)));

  MyObject deserialized = gson.fromJson("{ \"photo\": 12 }", MyObject.class);

  System.out.println(deserialized);
}

并且它会打印(注意我使用了自定义的toString()作为MyObject):

{"photo":"myUrl"}
MyObject{url='null', number=12}
jpfvwuh4

jpfvwuh42#

@SerializedName("photo")
private Object imageUrl;

public String getImageB64()
{
    return (String) this.imageUrl
}

public void setImageB64( String image )
{
    this.imageUrl = image
}

public ImageURL getimageURL()
{
    return (ImageURL) this.imageUrl
}

public void setimageURL( ImageURL imageUrl )
{
    this.imageUrl = imageUrl
}

相关问题