【Android - IPC】之Serializable和Parcelable序列化
1、序列化的目的
(1)永久的保存对象数据(将对象数据保存到文件或磁盘中);
(2)通过序列化操作将对象数据在网络上进行传输(由于网络传输是以字节流的方式对数据进行传输的,因此序列化的目的是将对象数据转化成字节流的形式);
(3)将对象数据在进程之间进行传递(Activity之间传递对象数据时,需要在当前的Activity中对对象数据进行序列化操作,在另一个Activity中需要进行反序列化操作将对象数据取出);
(4)Android中,使用Intent在Activity之间传递数据的时候,基本数据类型直接进行相关传递即可,但是一旦数据类型比较复杂的时候就不需要进行序列化操作了。
2、使用Serializable进行序列化
使用Serializable进行序列化的代码如下:
public class UserSerializable implements Serializable {
private static final long serialVersionUID = 1L;
public int userId;
public String userName;
public boolean isMale;
public UserSerializable(int userId, String userName, boolean isMale) {
this.userId = userId;
this.userName = userName;
this.isMale = isMale;
}
}
Serializable是Java所提供的一个序列化接口,它是一个空接口,为对象提供标准的序列化和反序列化的操作。使用Serializable来实现序列化相当简单,只需要在类的声明中指定一个 serialVersionUID 即可自动实现默认的序列化过程。
声明了类定义之后,我们可以通过ObjectOutputStream和ObjectInputStream来轻松实现序列化和反序列化操作,针对上面的 UserSerializable 类的操作代码如下:
// 序列化过程
UserSerializable user = new UserSerializable(1, "Jack", true);
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("cache.txt"));
out.writeObject(user);
out.close(); // 反序列化过程
ObjectInputStream in = new ObjectInputStream(new FileInputStream("cache.txt"));
UserSerializable newUser = (UserSerializable)in.readObject();
in.close();
注意:序列化的对象和反序列化出来的对象中的内容完全一样,但是两者并不是同一个对象。
下面来解释一下serialVersionUID的含义。serialVersionUID是用来辅助序列化和反序列化过程的,原则上序列化后的数据中的serialVersionUID只有和当前类的serialVersionUID相同才能够正常地被反序列化。
serialVersionUID的工作机制:序列化的时候,系统会把当前类的serialVersionUID写入序列化的文件中(也可能是其他中介),当反序列化的时候系统会去检测文件中的serialVersionUID,看它是否和当前类的serialVersionUID一致,如果一致就说明序列化的类的版本和当前类的版本是相同的,这个时候可以成功的反序列化;否则就说明当前类和序列化的类相比发生了某些改变(如成员变量的数量、类型发生改变),这个时候就无法正常的序列化了。
一般来说,我们可以自己显式地手动指定serialVersionUID的值(随便一个long类型的值即可),这样的话,不管类中的数据怎样变化,最终存储到文件(或其他中介)中的serialVersionUID的值都不会改变,因此可以最大限度的保证序列化和反序列化过程的成功;而如果不显式地指定serialVersionUID,则每次系统都会根据类的结构生成一个hash值并赋值给serialVersionUID,因此当类结构发生改变后,就可能造成反序列化过程失败。
注意:
(1)静态成员变量和用transient关键字修饰的变量不参与序列化过程;
(2)如果类结构发生了非常规性改变,比如修改了类名、修改了成员变量的类型,这个时候尽管serialVersionUID验证通过了,但是反序列化操作还是会失败,因为类结构有了毁灭性的改变;
(3)使用Serializable进行序列化是使用反射机制实现的,会产生很多临时变量;
(4)序列化过程知识针对变量进行序列化,不会针对方法进行序列化。
3、使用Parcelable进行序列化
使用Parcelable进行序列化的代码如下:
public class UserParcelable implements Parcelable {
public int userId;
public String userName;
public boolean isMale;
public Book book;
public UserParcelable(int userId, String userName, boolean isMale, Book book) {
this.userId = userId;
this.userName = userName;
this.isMale = isMale;
this.book = book;
}
// 从序列化后的对象中反序列化出原始对象
protected UserParcelable(Parcel in) {
this.userId = in.readInt();
this.userName = in.readString();
this.isMale = in.readInt() == 1;
this.book = in.readParcelable(Thread.currentThread().getContextClassLoader());
}
// 反序列化
public static final Creator<UserParcelable> CREATOR = new Creator<UserParcelable>() {
@Override
public UserParcelable createFromParcel(Parcel in) {
return new UserParcelable(in);
}
@Override
public UserParcelable[] newArray(int size) {
return new UserParcelable[size];
}
};
// 返回当前对象的内容描述,几乎在所有情况下都返回 0
@Override
public int describeContents() {
return 0;
}
// 序列化,将当前对象写入到序列化结构中,flags 参数在几乎所有情况下都为 0
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(userId);
dest.writeString(userName);
dest.writeInt(isMale ? 1 : 0);
dest.writeParcelable(book, 0);
}
}
Parcelable也是一个接口,只要实现这个接口,一个类的对象就可以实现序列化并可以通过Intent和Binder传递。
从上面的示例代码中可以看到, writeToParcel() 方法用来进行序列化; Creator<UserParcelable> 用来反序列化(可以反序列化成对象或对象数组); describeContents() 方法用来进行内容描述(一般都会返回0)。
在上面的代码中,UserParcelable类中除了int、String、boolean三种基本数据类型之外,还有一个自定义的bean类Book,如果想要对Book对象也进行序列化和反序列化,则Book类必须也实现Parcelable接口,且在序列化和反序列化的时候,调用 readParcelable() 方法和 writeParcelable() 方法。
Parcelable的工作原理:
(1)在Parcelable的操作中,无论读还是写,都需要使用Parcel作为中间层进行数据传递;
(2)Parcel涉及到的东西和C++底层有关,整个读写全是在内存中进行,主要通过 malloc() 、 realloc() 、 memcpy() 等内存操作进行,因此效率会比使用Serializable高很多。
注意:使用Parcelable可以保证反序列化出来的对象就是原来的对象,而不是重新new出来的对象。
4、两种方法的比较
(1)Serializable是Java中的序列化接口,而Parcelable是Android中的序列化接口;
(2)Serializable使用了反射机制,在序列化操作的时候会产生大量的临时变量,从而导致GC的频繁调用;Parcelable是以IBinder作为信息载体的,在内存上开销比较小,因此在内存的使用和性能方面,Parcelable会稍强于Serializable;
(3)在读写数据的时候,Serializable是通过I/O流的形式在磁盘上进行数据的读写,而Parcelable则是在内存中进行数据的读写;
(4)虽然使用Parcelable比使用Serializable麻烦,但它的效率很高,因此在很多情况下都推荐使用Parcelable;
(5)在不同的Android版本中,Parcelable可能会不同,因此Parcelable可能不能将数据持久化,也就是说,如果要想对数据进行持久化操作,还是需要使用Serializable。
【Android - IPC】之Serializable和Parcelable序列化的更多相关文章
- Android中Serializable和Parcelable序列化对象详解
学习内容: 1.序列化的目的 2.Android中序列化的两种方式 3.Parcelable与Serializable的性能比较 4.Android中如何使用Parcelable进行序列化操作 5.P ...
- Android开发之Serializable 和 Parcelable的差别(源码分享)
android 中自己定义的对象序列化的问题有两个选择一个是Parcelable,另外一个是Serializable. 一 序列化原因: 1.永久性保存对象.保存对象的字节序列到本地文件里. 2.通过 ...
- android ipc通信机制之二序列化接口和Binder
IPC的一些基本概念,Serializable接口,Parcelable接口,以及Binder.此核心为最后的IBookManager.java类!!! Serializable接口,Parcelab ...
- Android序列化之Serializable和Parcelable
PS:还有几天就开学了.先来一发. 学习内容: 1.序列化的目的 2.Android中序列化的两种方式 3.Parcelable与Serializable的性能比较 4.Android中如何使用Par ...
- android Activity之间数据传递 Parcelable和Serializable接口的使用
Activity之间传数据时,为了避免麻烦,往往会将一些值封装成对象,然后将整个对象传递过去.传对象的时候有两种情况,一种是实现Parcelable接口,一种是实现Serializable接口.0.解 ...
- android中实现Parcelable序列化步骤
import java.io.Serializable; import java.text.DecimalFormat; import android.os.Parcel; import androi ...
- Android中Parcelable序列化总结
在使用Parcelable对android中数据的序列化操作还是比较有用的,有人做过通过对比Serializable和Parcelable在android中序列化操作对象的速度比对,大概Parcela ...
- 浅谈Android中Serializable和Parcelable使用区别
版权声明:本文出自汪磊的博客,转载请务必注明出处. 一.概述 Android开发的时候,我们时长遇到传递对象的需求,但是我们无法将对象的引用传给Activity或者Fragment,我们需要将这些对象 ...
- 序列化Serializable和Parcelable
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 简单记录下序列化Serializable和Parcelable的使用方法. Android中Intent如果要传递类对象,可以通过两 ...
随机推荐
- C和C++引用传递和数组传参引用
引用传递有两种传参方式,具体可参考文章 概括地讲,就是 *声明一个形参是指针,所以需要传递指针实参,对应的函数实现也应当遵循指针的语法.这种实现思路并不针对于C或者C++,因为它们都有指针,所以都可以 ...
- Coremail接口存配置读取漏洞POC
Coremail产品诞生于1999年,经过二十多年发展,如今从亿万级别的运营系统,到几万人的大型企业,都有了Coremail的客户. 截止2019年,Coremail邮件系统产品在国内已拥有10亿终端 ...
- 洛谷P5522 【[yLOI2019] 棠梨煎雪】
区间操作考虑用线段树维护. 建\(n*2\)棵线段树,前\(n\)棵线段树维护每个串的第i位是否是0. 后\(n\)棵线段树维护每个串的第i位是否是1. 如果是问号的话,直接跳过就好(通过1和0能看出 ...
- C++学习笔记10_输入输出流.文件读写
//从键盘输入到程序,叫标准input:从程序输出到显示器,叫标准output:一并叫标准I/O //文件的输入和输出,叫文件I/O cout<<"hellow word&quo ...
- C++学习笔记1_ 指针.引用
1.引用的本质struct typeA{ int &a;}struct typeB{ int *a;}int main(void){ cout<<sizeof(struct typ ...
- Logback MDC
Mapped Diagnostic Contexts (MDC) (译:诊断上下文映射) Logback的设计目标之一是审计和调试复杂的分布式应用程序.大多数实际的分布式系统需要同时处理来自多个客 ...
- 「刷题」卡特兰数&prufer序列
1.网格 转换模型,翻折容斥出解. 2.有趣的数列 抽象一下模型,把奇数项当作横坐标,偶数项当作纵坐标,就是从n*n矩阵左下角走到右上角并且每一步x<=y的方案数,发现是卡特兰数,关于gcd,可 ...
- EffectiveJava-2
一.使用类库 使用类库的好处: 无须关心方法是如何实现的,由算法专家花了大量时间设计.实现和测试这个方法,不仅保证了正确性,而且一旦有缺陷,下一个版本就会修复. 不必浪费时间为哪些与工作不太相关的问题 ...
- 使用Typescript重构axios(十八)——请求取消功能:总体思路
0. 系列文章 1.使用Typescript重构axios(一)--写在最前面 2.使用Typescript重构axios(二)--项目起手,跑通流程 3.使用Typescript重构axios(三) ...
- 考试T1护花
传送门 这题的提议似乎有什么问题,只要约翰选好了要抓那头牛,他就不会吃草了,站在原地傻等? 这题就是贪心,但在用cmp中比较单位时间吃草数量时,要用double型,不然可能会有点一样... 还有就是主 ...