Android - GSon + RetroFit 中的继承和抽象 类
Android - Inheritance and abstract classes in GSon + RetroFit
我有以下 class 层次结构
public abstract class SyncModel {
@Expose
@SerializedName("id")
private Long globalId;
@Expose
protected DateTime lastModified;
/* Constructor, methods... */
}
public class Event extends SyncModel {
@Expose
private String title;
/* Other fields, constructor, methods... */
}
我需要发送一个事件实例到后端。
案例一。@Body
当我 post 请求正文中的事件实例时,它可以很好地序列化。
RetroFit Java 接口:
public interface EventAPI {
@POST("/event/create")
void sendEvent(@Body Event event, Callback<Long> cbEventId);
}
RetroFit 日志:
D Retrofit ---> HTTP POST http://hostname:8080/event/create
D Retrofit Content-Type: application/json; charset=UTF-8
D Retrofit Content-Length: 297
D Retrofit {"title":"Test Event 01",...,"id":null,"lastModified":"2015-07-09T14:17:08.860+03:00"}
D Retrofit ---> END HTTP (297-byte body)
案例二。@Field
但是当我post请求参数中的Event实例时,只有抽象class被序列化。
RetroFit Java 接口:
@FormUrlEncoded
@POST("/event/create")
void sendEvent(@Field("event") Event event, Callback<Long> cbEventId);
RetroFit 日志:
D Retrofit ---> HTTP POST http://hostname:8080/event/create
D Retrofit Content-Type: application/x-www-form-urlencoded; charset=UTF-8
D Retrofit Content-Length: 101
D Retrofit event=SyncModel%28globalId%3Dnull%2C+lastModified%3D2015-07-09T13%3A36%3A33.510%2B03%3A00%29
D Retrofit ---> END HTTP (101-byte body)
注意区别。
问题
为什么?
如何在请求参数中将序列化的事件实例发送到后端?
我是否需要为抽象 class 编写自定义 JSON 序列化程序? (示例:Polymorphism with JSON)
或者它是 RetroFit 的特定功能(忽略子 classes)?
我也注意到第二种情况globalId
字段序列化名称是globalId
,但应该是id
!这让我觉得 RetroFit 对 @Field
使用的 GsonConverter
与 @Body
参数不同...
配置
Gradle 依赖关系
compile 'com.squareup.retrofit:retrofit:1.9.+'
compile 'com.squareup.okhttp:okhttp:2.3.+'
compile 'net.danlew:android.joda:2.8.+'
compile ('com.fatboyindustrial.gson-jodatime-serialisers:gson-jodatime-serialisers:1.1.0') { // GSON + Joda DateTime
exclude group: 'joda-time', module: 'joda-time'
}
REST 客户端
public final class RESTClient {
// Not a real server URL
public static final String SERVER_URL = "http://hostname:8080";
// one-time initialization
private static GsonBuilder builder = new GsonBuilder()
.serializeNulls()
.excludeFieldsWithoutExposeAnnotation()
.setDateFormat("yyyy'-'MM'-'dd'T'HH':'mm':'ss'.'SSS'Z'");
// Joda DateTime type support
private static Gson gson = Converters.registerDateTime(builder).create();
private static RestAdapter restAdapter = new RestAdapter.Builder()
.setLogLevel(RestAdapter.LogLevel.FULL) // for development
.setEndpoint(SERVER_URL)
.setConverter(new GsonConverter(gson)) // custom converter
.build();
private static final EventAPI eventService = restAdapter.create(EventAPI.class);
/* + Getter for eventService */
static {
// forget them
restAdapter = null;
gson = null;
builder = null;
}
}
致电
RESTClient.getEventService().sendEvent(event, new Callback<Long>() {/* ... */});
看看@Field
's documentation。它说:
Values are converted to strings using String#valueOf(Object)
and then form URL encoded.
String#valueOf(Object)
在内部调用 Object#toString()
。我想您的 SyncModel
有一个 toString()
方法而 Event
没有。当 Retrofit 调用 String.valueOf(event)
时,会调用 SyncModel#toString()
而不是 Event#toString()
。这就是为什么您在 Retrofit 日志中看不到 title
的原因。
Gson 在转换@Field
参数时根本不起任何作用。可以 - 你可以让你的 toString()
方法看起来像这样:
@Override
public String toString() {
return GsonProvider.getInstance().toJson(this);
}
把它放在你的摘要中 SyncModel
class 它应该也适用于 Event
。
我有以下 class 层次结构
public abstract class SyncModel {
@Expose
@SerializedName("id")
private Long globalId;
@Expose
protected DateTime lastModified;
/* Constructor, methods... */
}
public class Event extends SyncModel {
@Expose
private String title;
/* Other fields, constructor, methods... */
}
我需要发送一个事件实例到后端。
案例一。@Body
当我 post 请求正文中的事件实例时,它可以很好地序列化。
RetroFit Java 接口:
public interface EventAPI {
@POST("/event/create")
void sendEvent(@Body Event event, Callback<Long> cbEventId);
}
RetroFit 日志:
D Retrofit ---> HTTP POST http://hostname:8080/event/create D Retrofit Content-Type: application/json; charset=UTF-8 D Retrofit Content-Length: 297 D Retrofit {"title":"Test Event 01",...,"id":null,"lastModified":"2015-07-09T14:17:08.860+03:00"} D Retrofit ---> END HTTP (297-byte body)
案例二。@Field
但是当我post请求参数中的Event实例时,只有抽象class被序列化。
RetroFit Java 接口:
@FormUrlEncoded
@POST("/event/create")
void sendEvent(@Field("event") Event event, Callback<Long> cbEventId);
RetroFit 日志:
D Retrofit ---> HTTP POST http://hostname:8080/event/create D Retrofit Content-Type: application/x-www-form-urlencoded; charset=UTF-8 D Retrofit Content-Length: 101 D Retrofit event=SyncModel%28globalId%3Dnull%2C+lastModified%3D2015-07-09T13%3A36%3A33.510%2B03%3A00%29 D Retrofit ---> END HTTP (101-byte body)
注意区别。
问题
为什么?
如何在请求参数中将序列化的事件实例发送到后端?
我是否需要为抽象 class 编写自定义 JSON 序列化程序? (示例:Polymorphism with JSON)
或者它是 RetroFit 的特定功能(忽略子 classes)?
我也注意到第二种情况globalId
字段序列化名称是globalId
,但应该是id
!这让我觉得 RetroFit 对 @Field
使用的 GsonConverter
与 @Body
参数不同...
配置
Gradle 依赖关系
compile 'com.squareup.retrofit:retrofit:1.9.+'
compile 'com.squareup.okhttp:okhttp:2.3.+'
compile 'net.danlew:android.joda:2.8.+'
compile ('com.fatboyindustrial.gson-jodatime-serialisers:gson-jodatime-serialisers:1.1.0') { // GSON + Joda DateTime
exclude group: 'joda-time', module: 'joda-time'
}
REST 客户端
public final class RESTClient {
// Not a real server URL
public static final String SERVER_URL = "http://hostname:8080";
// one-time initialization
private static GsonBuilder builder = new GsonBuilder()
.serializeNulls()
.excludeFieldsWithoutExposeAnnotation()
.setDateFormat("yyyy'-'MM'-'dd'T'HH':'mm':'ss'.'SSS'Z'");
// Joda DateTime type support
private static Gson gson = Converters.registerDateTime(builder).create();
private static RestAdapter restAdapter = new RestAdapter.Builder()
.setLogLevel(RestAdapter.LogLevel.FULL) // for development
.setEndpoint(SERVER_URL)
.setConverter(new GsonConverter(gson)) // custom converter
.build();
private static final EventAPI eventService = restAdapter.create(EventAPI.class);
/* + Getter for eventService */
static {
// forget them
restAdapter = null;
gson = null;
builder = null;
}
}
致电
RESTClient.getEventService().sendEvent(event, new Callback<Long>() {/* ... */});
看看@Field
's documentation。它说:
Values are converted to strings using
String#valueOf(Object)
and then form URL encoded.
String#valueOf(Object)
在内部调用 Object#toString()
。我想您的 SyncModel
有一个 toString()
方法而 Event
没有。当 Retrofit 调用 String.valueOf(event)
时,会调用 SyncModel#toString()
而不是 Event#toString()
。这就是为什么您在 Retrofit 日志中看不到 title
的原因。
Gson 在转换@Field
参数时根本不起任何作用。可以 - 你可以让你的 toString()
方法看起来像这样:
@Override
public String toString() {
return GsonProvider.getInstance().toJson(this);
}
把它放在你的摘要中 SyncModel
class 它应该也适用于 Event
。