美文网首页
IPC之binder

IPC之binder

作者: JasonChen8888 | 来源:发表于2020-02-27 10:06 被阅读0次

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

    相关文章

      网友评论

          本文标题:IPC之binder

          本文链接:https://www.haomeiwen.com/subject/uuuumctx.html