[原创]如何在Parcelable中使用泛型

实体类在实现Parcelable接口时,除了要实现它的几个方法之外,还另外要定义一个静态常量CREATOR,如下例所示:

     public static class ProductModel implements Parcelable {
public String id;
public String title;
public String sold_num;
public String max_month;
public String cprice;
public double month_price;
public List<String> gpic; @Override
public int describeContents() {
return 0;
} @Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(this.id);
dest.writeString(this.title);
dest.writeString(this.sold_num);
dest.writeString(this.max_month);
dest.writeString(this.cprice);
dest.writeDouble(this.month_price);
dest.writeStringList(this.gpic);
} public ProductModel() {
} protected ProductModel(Parcel in) {
this.id = in.readString();
this.title = in.readString();
this.sold_num = in.readString();
this.max_month = in.readString();
this.cprice = in.readString();
this.month_price = in.readDouble();
this.gpic = in.createStringArrayList();
} public static final Creator<ProductModel> CREATOR = new Creator<ProductModel>() {
public ProductModel createFromParcel(Parcel source) {
return new ProductModel(source);
} public ProductModel[] newArray(int size) {
return new ProductModel[size];
}
};

CREATOR在这里成为了一个约定,而没有放到接口定义里面,个人感觉这样封装得不是很好,不知道是不是实在没有更好的解决办法才弄成这样的?

假如在类里要使用泛型,麻烦就来了,例如这样

 public class PageDataModel<T extends Parcelable> extends APIDataModel {

     static final String DATA_KEY = "data";

     /**
* count : 1
* total_pages : 1
* list_rows : 10
* first_row : 0
*/ public int count;
public ArrayList<T> data;
public int total_pages;
public int list_rows;
public int first_row;
}

这个data,就无法正常read(write倒是可以)。

正常的write方式是这样的:

this.data = dest.createTypedArrayList(Parcelable.Creator<T> c)

那么问题来了,你只有一个泛型T,没有具体类型,拿不到它的CREATOR!CREATOR只是一个约定,而且是跟具体类型绑定的。

所以我前面说如果这个CREATOR能通过某种方式定义下来,这里或许就能拿到了,然而并没有。

但是android系统你是改变不了的,我们仍然要解决问题。经过多番调查尝试之后,终于找到了解决办法:

Pacel是可以读取和写入Bundle对象的,而Bundle对象又可以读取和写入Parcelable,而且不需要CREATOR(为啥Parcel不能设计成这样,坑爹呢。。)

那么 我们把泛型数据用Bundle包装一下即可,下面是具体代码,亲测有效,总算解决这个问题了。

 //write
Bundle bundle = new Bundle();
bundle.putParcelableArrayList(DATA_KEY, data);
dest.writeBundle(bundle); //read
this.data = in.readBundle().getParcelableArrayList(DATA_KEY);
04-26 13:20