本文参照《Android开发艺术探索》以及官方文档写的读书笔记,欢迎讨论,请勿转载。
在《Android开发艺术探索》这么书中使用跨进程通讯的两个场景:
1、通过多进程来获取多分内存空间
2、当前应用需要向其他应用获取数据
跨进程对我的理解就像请求网络调用接口似的,声明一个接口去在客户端调用,只不过调用的时候当前线程阻塞了,通过序列化的方式将用户自定义的对象(实现了Parcelable接口)或AIDL对象从客户端发送到服务端。然后服务端所对应的方法运行在一个Binder池内,将对象反序列化,从而得到对象,这时候将返回值返回,主线程从阻塞状态中重新恢复到运行的状态。
注意几点:
1、当非跨进程时(在Manifest没有指定process属性),当客户端调用AIDL方法时,客户端在UI线程,那么被调用的服务端的方法也在UI线程。反之。如果。客户端调用AIDL方法时,客户端在非UI线程,那么被调用的服务端方法也在非UI线程。
2、当跨进程时(在Manifest指定process属性),服务端的方法运行在Binder的线程池中,所以Binder方法不管是否耗时都应该采用同步的方式去实现。
3、当客户端发起远程请求时,由于当前线程会被挂起直至服务端进程返回数据,所以如果一个远程方法是很耗时的,那么不能再UI线程中发起此远程请求。
4、如果一个远程方法是耗时的,我在客户端调用时会被挂起(我上面说的第3点)。但是如果我的远程方法返回void,也就是远程方法在怎么耗时,我客户端也不需要你返回数据,那么可以在AIDL方法中增加oneway关键字。
oneway void addBook(in Book book)
这样客户端就不会被挂起了。
5、在跨进程中,客户端传递的对象是经过序列化的,在服务端接收的对象,是经过反序列化的。服务端接收的对象只是里面的东西一样,但是根本不是同一个对象,这点要注意。
6、所有非原语(原语含义看下面)参数都需要指示数据走向的方向标记。可以是 in、out 或 inout。
原语默认为 in,不能是其他方向。
7、AIDL接口中只支持方法,不支持声明静态常量。
刚刚说到类似调用接口,所以得新建一个AIDL接口文件。
package ipc;
import ipc.Book;
import ipc.IOnNewBookArrivedListener;
// Declare any non-default types here with import statements
interface IBookManager {
void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
double aDouble, String aString);
List<Book> getBookList();
void addBook(in Book book);
void registerListener(IOnNewBookArrivedListener listener);
}
在官网给的文档中说明AIDL文件支持的数据有:
Java 编程语言中的所有原语类型(如 int、long、char、boolean 等等)
String
CharSequence
List
Map
在《Android开发艺术探索》中作者加入了其他内容
Parcelable:所有实现了Parcelable接口的对象
AIDL:所有的AIDL接口本身也可以在AIDL文件中使用
其中自定义的Parcelable对象和AIDL对象要显示import进来,不管他们是否和当前的AIDL文件位于同一包内。
最后还要声明一个实现Parcelable对象的 .aidl 文件。 此 .aidl 文件与 C 语言中的头文件类似,并未编译。
// Book.aidl
package ipc;
parcelable Book;
// IOnNewBookArrivedListener.aidl
package ipc;
// Declare any non-default types here with import statements
import ipc.Book;
interface IOnNewBookArrivedListener {
void onNewBookArrivedListener(in Book newBook);
}
package ipc;
/**
* Created by apple on 2018/1/1.
*/
import android.os.Parcel;
import android.os.Parcelable;
public class Book implements Parcelable {
public int bookId;
public String bookName;
public Book(String bookName, int bookId) {
this.bookName = bookName;
this.bookId = bookId;
}
@Override
public String toString() {
return "Book{" +
"bookId=" + bookId +
", bookName='" + bookName + '\'' +
'}';
}
protected Book(Parcel in) {
bookId = in.readInt();
bookName = in.readString();
}
public static final Parcelable.Creator<Book> CREATOR = new Parcelable.Creator<Book>() {
@Override
public Book createFromParcel(Parcel in) {
return new Book(in);
}
@Override
public Book[] newArray(int size) {
return new Book[size];
}
};
public int getBookId() {
return bookId;
}
public void setBookId(int bookId) {