IPC基础概念介绍
IPC中的一些基础概念,主要包含三方面内容:Serializable
接口、Parcelable
接口和Binder
,只要熟悉这三方面的内容后,我们才能更好地理解跨进程通信的各种方式。Serializable
和Parcelable
接口可以完成对象的序列化过程,当我们需要通过Intent
和Binder
传输数据时就需要使用Serializable
或者Parcelable
。还有的时候我们需要把对象持久化到存储设备上或者通过网络传输给其他客户端,这个时候也需要使用Serializable
来完成对象持久化。
Serializable接口
Serializable
是Java所提供的一个序列化接口,它是一个空接口,为对象提供标准的序列化和反序列化操作。使用Serializable
来实现序列化相当简单,只需要这个类实现Serializable
接口并声明一个serialVersionUID
即可自动实现默认的序列化过程:
package com.wh.myapplication;
import java.io.Serializable;
public class User implements Serializable {
private static final long serialVersionUID = 519067123721295773L;
public int userId;
public String userName;
public boolean isMale;
public User(int userId, String userName, boolean isMale) {
this.userId = userId;
this.userName = userName;
this.isMale = isMale;
}
}
使用ObjectOutputStream
和ObjectInputStream
进行对象的序列化和反序列化:
//序列化过程
User user = new User(0, "jay", true);
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("cache.txt"));
out.writeObject(user);
out.close();
//反序列化过程
ObjectInputStream in = new ObjectInputStream(new FileInputStream("cache.text"));
User newUser = (User) in.readObject();
in.close();
上述代码演示了采用Serializable
方式序列化对象的典型过程,只需要把实现了Serializable
接口的User
对象写到文件中就可以快速恢复了,恢复后的对象newUser
和user
的内容完全一样,但是两者并不是同一个对象。
serialVersionUID
不是必须,可以实现序列化,但是会影响反序列化,因为serialVersionUID
的详细工作机制是这样的:序列化的时候系统会把当前类的serialVersionUID
写入序列化的文件中(也可能是其他的中介),当反序列化的时候,系统就会去检测文件中的serialVersionUID
,看它是否和当前类的serialVersionUID
一致,如果一致说明序列化的类的版本和当前类的版本相同,这个时候可以成功反序列化,否则说明当前类和序列化的类相比,发生了变换,比如成员变量的数量、类型可能发生了改变,这个时候就无法正常反序列化。序列化和反序列化之后不是同一个对象。这个时候是无法正常反序列化的,因此会报如下错误:
java.io.InvalidClassException: Main; local class incompatible: stream classdesc serialVersionUID = 8711368828010083044, local class serialVersionUID = 8711368828010083043。
系统默认序列化过程是可以改变的,通过实现如下两个方法即可重写系统默认的序列化和反序列化过程,不过大部分情况下我们不需要重写这两个方法。
private void writeObject(java.io.ObjectOutputStream out) throws IOException {
// write 'this' to 'out'...
}
private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException {
// populate the fields of 'this' from the data in 'in'...
}