Android IPC 之AIDL
来源:互联网 发布:低功耗51单片机 编辑:程序博客网 时间:2024/04/30 00:01
这几天深圳阴雨绵绵,让一个宅男连出去的欲望都没了。
一:概述
上一篇写了下Messenger的如何使用,这篇写一下AIDL 如何使用,Messenger是一对多的串行使用,而AIDL是可以一对多的并发。
看看效果图吧:
以打印log的方式出来的:
二:看看代码
在aidl中仅支持以下数据类型:
1.基本数据类型(int,long,char,boolean,byte等)2.String 和 CharSequence;3.List:只支持ArrayList,并且他其中的元素必须被aidl支持4.Map:只支持HashMap,里面的每个元素也需要被支持5.Parcelable:所有实现了这个接口的对象6.aidl:所有的aidl接口本身也可以在aidl文件中使用
其中自定义的parcelable对象和aidl对象必须要显示import进来,不管是否在同一个包中。
并且如果使用了book这个类,必须要新建一个同名的aidl文件,声明为parcelable类型。
不然Android studio中会报:
java.lang.runtimeExcepition: com.android.ide.common.process.processException:org.gradle.process.internal.execException:process ‘command ‘E:|androidsdk\sdk\build-toos\23.0.2\aidl.exe” finished with non-sero exit value 1
这个demo中我们使用了一个 IBookManager.aidl文件,然后其中内部还使用了一个Book类,当然这个Book类必须实现parcelable接口,这个aidl中需要新建一个Book.aidl文件,然后将这个Book声明为一个parcelable对象。
Book.aidl:
然后看看IBookManager.aidl中:
导入book,当然后面还使用了一个 IOnNewBookArrivedListener,用来做一个观察模式:
我们就看看
List getBookList();
void addBook(in Book book);
这两个方法
一个是客户端从服务端获取所有book,一个是客户端为服务端添加一本书的方法。
这两个方法运行在客户端,并且调用的时候当前的线程会挂起,然后直至方法运行至服务端运行完毕然后返回线程才继续运行,所以这两个方法如果运行在客户端的ui线程中的话那么是有anp(应用程序未响应)的危险的。所以最好还是另开一个线程去调用。
然后服务端的binder运行在binder线程池中,所以方法可以使用同步的方式,因为已经是运行在另个线程中了。
好现在我们新建一个Service,这个service需要是运行在另个进程中的。
<service android:name=".BookManagerService" android:enabled="true" android:exported="true" android:process=":remote" />
我们先看看两个集合:
private CopyOnWriteArrayList<Book> mBookList = new CopyOnWriteArrayList<Book>(); private RemoteCallbackList<IOnNewBookArrivedListener> listenerLists = new RemoteCallbackList<IOnNewBookArrivedListener>();
使用的CopyOnWriteArrayList来保存book,因为这个是自动的线程同步的。
RemoteCallbackList是用来专门保存进程的listener,因为用普通的ArrayList,或者CopyOnWriteArrayList,当客户端想去删除的某个listener的时候传入的listener和服务端的listener并不是同一个。所以删除会出现问题。但是为什么这个可以呢,因为用的底层binder直接使用的(因为底层的binder是同一个)。
然后看看我们实现IBookManager.aidl中的接口
private Binder mBinder = new IBookManager.Stub() { public List<Book> getBookList() { return mBookList; } public void addBook(Book book) { mBookList.add(book); } public void registerLintener(IOnNewBookArrivedListener lis) { listenerLists.register(lis); final int count = listenerLists.beginBroadcast(); Log.e("xhc", "listenerLists.size " + count); listenerLists.finishBroadcast(); } public void unregisterListener(IOnNewBookArrivedListener lis) { listenerLists.unregister(lis); final int count = listenerLists.beginBroadcast(); Log.e("xhc", "listenerLists.size " + count); listenerLists.finishBroadcast(); } };
就是实现了四个方法
注意listenerLists.beginBroadcast(); listenerLists.finishBroadcast();需要成对使用。
每次到了一本新书,就发送各个通知。
//新书到了, 添加一个观察者模式 private void onNewBookArrived(Book book) throws RemoteException { mBookList.add(book); final int count = listenerLists.beginBroadcast(); for (int i = 0; i < count; ++i) { IOnNewBookArrivedListener l = listenerLists.getBroadcastItem(i); if (l != null) { try { l.onNewBookArrived(book); } catch (Exception e) { } } } listenerLists.finishBroadcast(); }
最后我们来看看客户端如何实现的,Activity的oncreate中进行BindService
Intent intent = new Intent(this,BookManagerService.class); bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE);
获得IBookManager,然后远程调用各个方法,并且注册了一个通知。
private ServiceConnection serviceConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { bookManager = IBookManager.Stub.asInterface(service); try{ List<Book> list = bookManager.getBookList(); Log.e("xhc",list.toString()); Book book = new Book(3,"好好学习"); bookManager.addBook(book); Book book2 = new Book(4,"天天向上"); bookManager.addBook(book2); list = bookManager.getBookList(); Log.e("xhc",list.toString()); bookManager.registerLintener(iOnNewBookArrivedListener); }catch(Exception e){ e.printStackTrace(); } } };
在onDestory()中注销通知。注销的通知 iOnNewBookArrivedListener,和注册的通知是同一个,因为中间经历了一层binder,所以这个iOnNewBookArrivedListener不是服务端的iOnNewBookArrivedListener,所以服务端使用的 RemoteCallbackList来装。
@Override protected void onDestroy() { super.onDestroy(); try{ if(bookManager != null && bookManager.asBinder().isBinderAlive()){ bookManager.unregisterListener(iOnNewBookArrivedListener); } }catch(Exception e){ } unbindService(serviceConnection); }
刚睡醒。。。。。。。。。。无聊死了。。。。。要是深圳的童鞋们有谁想骑车玩的,拜托叫我一声啊。
源码下载
参考书籍《Android开发艺术探索》
加个好友共同学习(不是公众号):
因为小弟水平有限,如果有写的有问题,希望指出。
- Android IPC之AIDL
- Android IPC 之AIDL
- Android IPC之AIDL
- Android IPC 之 AIDL
- Android IPC之AIDL
- Android IPC之AIDL浅谈
- android IPC机制之 AIDL
- android进程间通信(IPC)之AIDL
- 【Android机制】IPC机制之AIDL
- Android IPC 之 AIDL(一)
- Android IPC 之 AIDL (二)
- Android进程间通信(IPC)之AIDL
- Android IPC机制之AIDL的使用
- Android IPC之AIDL使用解析
- Android IPC之AIDL的使用
- Android IPC通信之AIDL理解
- IPC之AIDL简析
- IPC通信之AIDL
- P18 (**) Extract a slice from a list.
- Apache常见编译选项-转
- 关于sqlite断电或者存储空间不足造成的数据文件损坏问题-转
- php 密码加密方式
- makefile how to
- Android IPC 之AIDL
- 安装cocoapods
- Bash中命令连接符的用法——一次执行多个命令-转
- Ubuntu 12.04中设置安装Google拼音输入法
- Ubuntu 13.10 用sogou拼音替换ibus-转
- org.testng.TestNGException: Cannot find class in classpath
- sqlite 按拼音排序
- UTF-8 编码里,一个汉字占用多少个字节 -转
- 机器学习笔记(一) : 线性建模——最小二乘法