Android中难免使用Intent在各个组件间进行传递数据,这个时候如果需要传递对象,就需要对对象进行序列化,序列化的方式有两种一种是parcelable,位于android.os包下,而serializable则位于java.io包下,下面分别对这两种方式进行一下分析。
一、通过parcelable方法实现序列化
notification.java 代码如下
public class Notification implements Parcelable {
private String title;
private String body;
private String icon;
private String sound;
private String tag;
private String color;
private String clickAction;
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getBody() {
return body;
}
public void setBody(String body) {
this.body = body;
}
public String getIcon() {
return icon;
}
public void setIcon(String icon) {
this.icon = icon;
}
public String getSound() {
return sound;
}
public void setSound(String sound) {
this.sound = sound;
}
public String getTag() {
return tag;
}
public void setTag(String tag) {
this.tag = tag;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
public String getClickAction() {
return clickAction;
}
public void setClickAction(String clickAction) {
this.clickAction = clickAction;
}
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(title);
dest.writeString(body);
dest.writeString(icon);
dest.writeString(sound);
dest.writeString(tag);
dest.writeString(color);
dest.writeString(clickAction);
}
static Parcelable.Creator
CREATOR = new Parcelable.Creator
() {
@Override
public Notification createFromParcel(Parcel source) {
return new Notification(source);
}
@Override
public Notification[] newArray(int size) {
return new Notification[size];
}
};
private Notification(Parcel source) {
title = source.readString();
body = source.readString();
;
icon = source.readString();
;
color = source.readString();
;
tag = source.readString();
;
clickAction = source.readString();
;
sound = source.readString();
}
public Notification() {
}
}
从代码中可以看出想完成序列化的工作有几点需要注意的。
1、必须实现parcelable接口,然后复写两个方法,一个是describeContents方法一个是writeToParcel方法,但是第一个describeContents方法我们基本上不用去理会他,关心的内容放在writeToParcel上。
2、特别需要注意的是除了复写两个方法后还需要新建一个静态的Parcelable.Creator类,这里需要注意的是类名必须是CREATOR,否则将会收到错误警告如图所示
3、最核心的操作就是对数据进行序列化只要通过两个方法来实现
通过这两个方法完成对数据的序列化操作,但是这里需要注意的是如果使用基本数据类型可以使用writeInt,writeLong,writeDobule等等方法,如果其中嵌套对象呢,看下面的Message.java代码
public class Message implements Parcelable {
private String to;
private String from;
private ArrayMap
data = new ArrayMap<>();
private String messageType;
private String messageId;
private long sendTime;
private Notification notification;
public String getMessageType() {
return messageType;
}
public void setMessageType(String messageType) {
this.messageType = messageType;
}
public String getMessageId() {
return messageId;
}
public void setMessageId(String messageId) {
this.messageId = messageId;
}
public long getSendTime() {
return sendTime;
}
public void setSendTime(long sendTime) {
this.sendTime = sendTime;
}
public String getTo() {
return to;
}
public void setTo(String to) {
this.to = to;
}
public String getFrom() {
return from;
}
public void setFrom(String from) {
this.from = from;
}
public ArrayMap
getData() {
return data;
}
public void setData(ArrayMap
data) {
this.data = data;
}
public Notification getNotification() {
return notification;
}
public void setNotification(Notification notification) {
this.notification = notification;
}
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(to);
dest.writeString(from);
dest.writeParcelable(notification, flags);
dest.writeLong(sendTime);
// dest.writeInt(data.size());
// for (Map.Entry
entry : data.entrySet()) {
// dest.writeString(entry.getKey());
// dest.writeString(entry.getValue());
// }
dest.writeMap(data);
}
static Creator
CREATOR = new Creator
() { @Override public Message createFromParcel(Parcel source) { return new Message(source); } @Override public Message[] newArray(int size) { return new Message[size]; } }; private Message(Parcel source) { to = source.readString(); from = source.readString(); notification = source.readParcelable(Notification.class.getClassLoader()); sendTime = source.readLong(); // int size = source.readInt(); // for (int i = 0; i < size; i++) { // String key = source.readString(); // String value = source.readString(); // data.put(key, value); // } source.readMap(data, ArrayMap.class.getClassLoader()); } public Message() { } }
代码中可以看出第一个message中嵌套一个实现了parcelable接口的类,上面我们提到了如果都是基本数据类型进行序列化,但是现在不是基本数据类型了我们如何序列化呢,想必在上面的message.java代码中已经看到了需要使用source.readParcelable(Notification.class.getClassLoader());通过readParcelable方法,这里需要注意的是参数需要一个classLoader,
千万不要传入null,否则序列化的时候会抛出异常。
既然有readParcelable那我们就需要对应的writeParcelable,通过这两个方法基本完成了对于实现parcelable对象嵌套的序列化。
3、对于嵌套map的序列化操作,上面我们提到了基本数据类型和对象的嵌套序列化,但是如果我现在用在类中嵌套一个map呢这个时候该如何序列化。答案还是在message.java中,我们通过下面方法实现
嵌套类的时候不需要初始化,map需要初始化,这一点我比较疑惑,如果是我姿势不对还请大家指点一二。
二、通过serializable方法实现序列化
这种方法比较容易实现,如下图所示
只需要通过实现serializable接口就可以了,如果里面还要嵌套类,只要将所有的类实现serializable接口就好了。
总结:通过上面的分析我们可以看出serializable使用简单,而parcelable稍微复杂一点,但是serializable的兼容性更好点,原因就是parcelable是android.os包下的,由于android系统版本差异,可能会有一些不同,但是在内容开销方面要比serializable少很多,所以效率和性能上会高于serializable,所以两种方式各有利弊选择使用哪个看自己的实际需求。
能想到的暂时就是这么多,希望有说的不对的地方大家指出来,大家一同进步,非常感谢。最后附上源码,方便大家查看。
http://download.csdn.net/detail/mahaiming1990/9566324