IPC之binder

2020-02-27  本文已影响0人  JasonChen8888

一、进程空间分配
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
结果。

上一篇下一篇

猜你喜欢

热点阅读