一、进程空间分配
1、一个进程空间分为 用户空间 & 内核空间(Kernel),即把进程内 用户 & 内核 隔离开来
2、二者区别:
a、进程间,用户空间的数据不可共享,即用户空间 = 不可共享空间
b、进程间,内核空间的数据可共享, 即内核空间 = 可共享空间
3、进程内 用户 与 内核 进行交互 称为系统调用
二、进程隔离
为了保证 安全性 & 独立性,一个进程 不能直接操作或者访问另一个进程,即Android的进程是相互独立、隔离的
三、跨进程通信(IPC)
1、跨进程间通信的原理
a、先通过 进程间 的内核空间进行 数据交互
b、再通过 进程间 的用户空间 & 内核空间进行 数据交互
IPC.png
以AIDL为例
代码:
package my.itgungnir.ipc.binder;
public interface IBookManager extends android.os.IInterface {
/**
* Local-side IPC implementation stub class.
*/
public static abstract class Stub extends android.os.Binder implements my.itgungnir.ipc.binder.IBookManager {
private static final java.lang.String DESCRIPTOR = "my.itgungnir.ipc.binder.IBookManager";
/**
* Construct the stub at attach it to the interface.
*/
public Stub() {
this.attachInterface(this, DESCRIPTOR);
}
/**
* Cast an IBinder object into an my.itgungnir.ipc.binder.IBookManager interface,
* generating a proxy if needed.
*/
public static my.itgungnir.ipc.binder.IBookManager asInterface(android.os.IBinder obj) {
if ((obj == null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if (((iin != null) && (iin instanceof my.itgungnir.ipc.binder.IBookManager))) {
return ((my.itgungnir.ipc.binder.IBookManager) iin);
}
return new my.itgungnir.ipc.binder.IBookManager.Stub.Proxy(obj);
}
@Override
public android.os.IBinder asBinder() {
return this;
}
@Override
public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException {
switch (code) {
case INTERFACE_TRANSACTION: {
reply.writeString(DESCRIPTOR);
return true;
}
case TRANSACTION_getBook: {
data.enforceInterface(DESCRIPTOR);
my.itgungnir.ipc.binder.Book _result = this.getBook();
reply.writeNoException();
if ((_result != null)) {
reply.writeInt(1);
_result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
} else {
reply.writeInt(0);
}
return true;
}
}
return super.onTransact(code, data, reply, flags);
}
private static class Proxy implements my.itgungnir.ipc.binder.IBookManager {
private android.os.IBinder mRemote;
Proxy(android.os.IBinder remote) {
mRemote = remote;
}
@Override
public android.os.IBinder asBinder() {
return mRemote;
}
public java.lang.String getInterfaceDescriptor() {
return DESCRIPTOR;
}
@Override
public my.itgungnir.ipc.binder.Book getBook() throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
my.itgungnir.ipc.binder.Book _result;
try {
_data.writeInterfaceToken(DESCRIPTOR);
mRemote.transact(Stub.TRANSACTION_getBook, _data, _reply, 0);
_reply.readException();
if ((0 != _reply.readInt())) {
_result = my.itgungnir.ipc.binder.Book.CREATOR.createFromParcel(_reply);
} else {
_result = null;
}
} finally {
_reply.recycle();
_data.recycle();
}
return _result;
}
}
static final int TRANSACTION_getBook = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
}
public my.itgungnir.ipc.binder.Book getBook() throws android.os.RemoteException;
}
最外层的getBook():一个抽象方法,就是我们在IBookManager.aidl文件中声明的方法;
TRANSACTION_getBook:一个整形的ID,用于表示客户端请求的是哪个方法;
DESCRIPTOR:Binder的唯一标识,一般用当前Binder的包路径表示;
asInterface():判断当前进程是服务端进程还是客户端进程,如果是服务端进程则返回Stub对象,否则返回Stub.Proxy对象;
asBinder():返回当前的Binder对象;
onTransact(int code, Parcel data, Parcel reply, int flag):这个方法运行在服务端的Binder线程池中,当客户端发起请求时,
就由这个方法来处理请求。服务端通过code获取客户端想要访问的目标方法;通过data来获取目标方法所需的参数;执行完
目标方法后,将返回值写入到reply中。另外,如果这个方法返回false,则客户端请求失败,我们可以通过这一点来判断客户
端是否有权访问我们的服务;
Proxy#getBook():这个方法运行在客户端,其内部实现是这样的:首先创建三个对象,_data用来存储这个方法的参数信息;
_reply用来存储从服务端返回的数据;_result用来作为返回值返回,然后调用transact()方法发起RPC(远程过程调用)请求,
调用服务端的onTransact()方法,同时当前线程挂起;当RPC过程返回后,当前线程继续执行,经过一系列处理后返回_result
结果。
网友评论