美文网首页
IPC(五)--AIDL使用进阶

IPC(五)--AIDL使用进阶

作者: azu_test | 来源:发表于2019-02-21 17:01 被阅读0次

移步Android跨进程通信IPC

简介

  1. Binder运行在服务端进程,如果服务端由于某种原因异常终止,这时候服务端和客户端之间的Binder连接会断裂也就是Binder死亡。这个时候如果再做远程调用就会失败,进而会导致客户端的功能异常。我们可以给Binder设置一个死亡代理,当Binder死亡时我们会收到通知回调,在回调内我们可以做重连操作以便恢复连接。
  2. 服务端直接调用客户端内方法获取数据,以便客户端和服务端双向交互。
  3. 做一个观察者模式,可以在服务端数据发生数据变化时及时通知客户端。

1. 设置死亡代理

主要有两个操作步骤,一是声明死亡代理,并在Binder的死亡回调中做重连等相关操作;二是设置死亡代理。

1.1 声明死亡代理
    //实例化死亡代理接口,并实现死亡回调的接口
    IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {
         //死亡回调的接口
        @Override
        public void binderDied() {
            if (mRemoteService == null){
                return;
            }
            //解绑死亡代理
            mRemoteService.asBinder().unlinkToDeath(mDeathRecipient,0);
            mRemoteService = null;
            // TODO: 做重新绑定service服务,或者其他的操作
            ...
        }
    };

TODO部分自行做操作,可以做和服务端的重连操作也可以做其他操作

1.2 设置死亡代理
    private ServiceConnection mConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName componentName, IBinder serverBinder) {
            mRemoteService = IService.Stub.asInterface(serverBinder);
            try {
                //给binder设置死亡代理
                serverBinder.linkToDeath(mDeathRecipient,0);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName componentName) {

        }
    };

上面部分的主要代码前文中已经做过解读,此处不再赘述。里面有一个额外的方法调用就是serverBinder.linkToDeath(),这个方法就是给Binder设置死亡代理,当Binder异常死亡时就会调用mDeathRecipient内的binderDied方法做死亡通知。

2. 设置客户端和服务端双向交互

做双向交互要做的核心就是客户端也要创建一个Binder对象并将Binder对象交付给服务端。其中观察者模式使用的核心也是如此。

2.1 客户端内创建Binder对象

创建回调使用的aidl

package com.wuyazhou.learn.IPC.aidl;
import com.wuyazhou.learn.IPC.aidl.Book;
interface ICallback {
    void callback(in Book newBook);
}

客户端内创建Binder对象

    private ICallback mCallback = new ICallback.Stub() {
        @Override
        public void callback(Book newBook) throws RemoteException {
            Log.d("wuyazhouAIDL","回调:"+newBook.bookName);
        }
    };

这里的Binder对象mCallback运行在客户端

2.2 将客户端内创建的Binder对象交付给服务端
    mRemoteService.useCallback(mCallback);
2.3 服务端调用客户端接口
    private final IService.Stub mBinder = new IService.Stub() {
        ...
        @Override
        public void useCallback(ICallback callback) throws RemoteException {
            callback.callback(new Book(456,"Android进阶"));
        }
    };

这里可能会疑惑为什么ICallback可以直接调用callback接口,并且能调用到客户端内实现的ICallback.aidl内的接口。
因为上一步我们通过Binder的传递机制将Binder对象ICallback传递给了服务端,所以服务端持有ICallback的Binder。

3. 观察者模式

观察者模式和上面的双向交互几乎一致,只是向多个客户端分发和解绑操作,所以这里要有一个存储向客户端回调的接口的容器。因为客户端将aidl的接口传递给Server时Server端会重新生成一个全新的对象,也就是说解绑时我们传递要解绑的aidl接口也会生成一个全新的对象,这两次生成的全新对象肯定是不一样的对象,所以普通的容器不能满足对aidl接口的对应增删。
RemoteCallbackList是系统专门提供的用于删除跨进程listener的接口。它的内部有一个Map用来专门存储AIDL回调。其中key是IBinder类型,value是Callback类型。

3.1 创建监听使用的aidl
package com.wuyazhou.learn.IPC.aidl;
import com.wuyazhou.learn.IPC.aidl.Book;
interface IOnNewBookArrivedListener {
    void onNewBookArrived(in Book newBook);
}
3.2 服务端接收和移除监听
    private final IService.Stub mBinder = new IService.Stub() {
        /** 添加监听*/
        @Override
        public void registerListener(IOnNewBookArrivedListener listener) {
            mRemoteCallbackList.register(listener);

            final int N = mRemoteCallbackList.beginBroadcast();
            mRemoteCallbackList.finishBroadcast();
        }

        /** 移除监听*/
        @Override
        public void unRegisterListener(IOnNewBookArrivedListener listener) {
            mRemoteCallbackList.unregister(listener);

            final int N = mRemoteCallbackList.beginBroadcast();
            mRemoteCallbackList.finishBroadcast();
        }
    };
3.3 客户端实现监听
    private IOnNewBookArrivedListener mOnNewBookArrivedListener = new IOnNewBookArrivedListener.Stub() {
        @Override
        public void onNewBookArrived(Book newBook) throws RemoteException {
            Log.d("wuyazhouAIDL","有新书到了");
        }
    };
3.4 客户端绑定监听
         mRemoteService.registerListener(mOnNewBookArrivedListener);
3.5 服务端分发监听
    private void onNewBookArrived(Book book) throws RemoteException {
        final int N = mRemoteCallbackList.beginBroadcast();
        for (int i = 0; i < N; i++) {
            IOnNewBookArrivedListener l = mRemoteCallbackList.getBroadcastItem(i);
            if (l != null) {
                try {
                    l.onNewBookArrived(book);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        }
        mRemoteCallbackList.finishBroadcast();
    }
    onNewBookArrived(new Book(123,"艺术探索"));

相关文章

  • IPC(五)--AIDL使用进阶

    移步Android跨进程通信IPC 简介 Binder运行在服务端进程,如果服务端由于某种原因异常终止,这时候服务...

  • 安卓AIDL使用详解

    安卓AIDL使用详解 说到AIDL就会提到IPC,具体说明是AIDL和IPC,这些概念大家自行查阅,在这里我就不多...

  • Android中的AIDL

    Android中的AIDL使用详解 AIDL使用解析 Android进阶之AIDL的使用详解

  • 016 Messenger

    除了使用 AIDL 进行 IPC 外,我们还可以使用 Messenger 来替代 AIDL。通过在 Message...

  • 底层学习 | Android IPC机制(四)Messenger

    一、概述 除了使用 AIDL 进行 IPC 外,我们还可以使用 Messenger 来替代 AIDL。通过在 Me...

  • AndroidIPC机制(4)-Messenger

    一、概述 除了使用 AIDL 进行 IPC 外,我们还可以使用 Messenger 来替代 AIDL。通过在 Me...

  • Android-服务(AIDL通讯)

    本文学习目标 学会使用AIDL与远程服务进行通讯 AIDL介绍 AIDL是Android中IPC(Inter-Pr...

  • 进程间通信(应用之间通信)

    进程间通信(IPC)方式 使用Bundle 使用文件共享 使用Messenger 使用AIDL 使用COntent...

  • 安卓实现IPC(三)—— AIDL

    继使用Intent实现IPC后,这篇文章来学习以下使用AIDL实现IPC,用Intent实现的方法可以查看上一篇,...

  • IPC之AIDL进阶用法

    接上篇笔记 AIDL入门用法 主要包括下边四点: 跨进程的接口回调【观察者】 线程问题 断开重连问题 权限校验问题...

网友评论

      本文标题:IPC(五)--AIDL使用进阶

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