Aidl实现进程间通信,跨进程回调
aidl支持的文件类型
1. Java 编程语言中的基本类型, 如 int、long、boolean 等, 不需要 import.
2. String、List、Map 和 CharSequence, 不需要 import.
3. AIDL 生成的 interface, 需要 import, 同一个包中也需要导入, 传递的是引用.
如果定义的接口方法中有参数, 则需在前面加上 in, out 或 inout, 但是对于基本类型的参数, 默认就是 in, 并且不会是其他值.
4. Parcelable 接口的自定义类, 需要 import, 同一个包中也需要导入, 其传递的是值.
另外, 自定义类还需要用一个aidl文件来声明该类型.
如定义了一个类, 它的源代码在RoadLine.java中:
package com.communicate;
import android.os.Parcel;
import android.os.Parcelable;
public class RoadLine implements Parcelable {
需要在RoadLine.aidl文件中做如下声明:
package com.communicate;
import com.communicate.RoadLine;
parcelable RoadLine;
AIDL 实现回调
1. 建立2个aidl文件, 一个用于声明回调的接口, 另一个用于定义注册接口, 及服务端调用的通知方法. 如:
package com.apical.contact; import com.apical.contact.RoadLine;
import com.apical.contact.CarData; interface IDVRRoadMetaDataCallback {
void onMetaData(int cameraId, in List<RoadLine> roadLines, in List<CarData> carDatas);
}
package com.apical.contact; import com.apical.contact.IDVRRoadMetaDataCallback;
import com.apical.contact.RoadLine;
import com.apical.contact.CarData; interface IDVRRoadMetaDataTaskBinder {
void registerCallback(int cameraId, IDVRRoadMetaDataCallback callback); void unregisterCallback(int cameraId, IDVRRoadMetaDataCallback callback); void onUpdateMetaData(int cameraId, in List<RoadLine> roadLines, in List<CarData> carDatas);
}
2. 实现服务端后台Service类. 要实现回调, 需要使用RemoteCallbackList对象, 一个存储回调对象的列表
private RemoteCallbackList<IDVRRoadMetaDataCallback> mCallbackList = new RemoteCallbackList<>();
注册
mCallbackList.register(callback);
取消注册
mCallbackList.unregister(callback);
方法调用
int count = mCallbackList.beginBroadcast();
try {
for (int i = 0; i < count; i++) {
mCallbackList.getBroadcastItem(i).onMetaData(cameraId, roadLines, carDatas);
}
} catch (RemoteException e) {
Dlog.e("E/onUpdateMetaDatas", e);
} finally {
mCallbackList.finishBroadcast();
}
package com.communicate; import java.util.List; import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteCallbackList;
import android.os.RemoteException;
import android.util.Log; public class DvrMetaDataService extends Service { private static final String TAG = "debug"; private RemoteCallbackList<IDVRRoadMetaDataCallback> mCallbackList = new RemoteCallbackList<>(); @Override
public IBinder onBind(Intent intent) {
return mBinder;
} @Override
public boolean onUnbind(Intent intent) {
return super.onUnbind(intent);
} @Override
public void onCreate() {
super.onCreate();
} @Override
public void onDestroy() {
super.onDestroy();
} public IDVRRoadMetaDataTaskBinder.Stub mBinder = new IDVRRoadMetaDataTaskBinder.Stub() { @Override
public void registerCallback(int cameraId, IDVRRoadMetaDataCallback callback) throws RemoteException {
mCallbackList.register(callback);
} @Override
public void unregisterCallback(int cameraId, IDVRRoadMetaDataCallback callback) throws RemoteException {
mCallbackList.unregister(callback);
} @Override
public void onUpdateMetaData(int cameraId, List<RoadLine> roadLines, List<CarData> carDatas) throws RemoteException {
int count = mCallbackList.beginBroadcast();
try {
for (int i = 0; i < count; i++) {
mCallbackList.getBroadcastItem(i).onMetaData(cameraId, roadLines, carDatas);
}
} catch (RemoteException e) {
Log.e(TAG, "E/onUpdateMetaDatas", e);
} finally {
mCallbackList.finishBroadcast();
}
}
};
}
注册服务,必须指定exported为true
<service android:name="com.communicate.DvrMetaDataService" android:exported="true">
</service>
3. 服务端绑定服务, 调用数据通知方法.
Intent metaDataService = new Intent();
metaDataService.setComponent(new ComponentName(com.communicate, com.communicate.DvrMetaDataService));
boolean result = bindService(metaDataService, mMetaDataConn, Service.BIND_AUTO_CREATE); /** DVR Meta data 共享数据服务 */
private IDVRRoadMetaDataTaskBinder mMetaDataServiceTaskBinder;
private ServiceConnection mMetaDataConn = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName componentName) {
} @Override
public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
mMetaDataServiceTaskBinder = IDVRRoadMetaDataTaskBinder.Stub.asInterface(iBinder);
}
}; 在更新数据时调用
if (mMetaDataServiceTaskBinder != null) {
try {
mMetaDataServiceTaskBinder.onUpdateMetaData(cameraId, roadLines, carDatas);
} catch (RemoteException e) {
Dlog.e("E/OnCameraMetaDataChange", e);
}
}
4. 客户端绑定服务, 注册接口, 实现回调的功能.需要注意的是, 由于服务端也绑定了服务,或者多个客户端同时绑定服务, unbindService时并不会调用 onServiceDisconnected 方法, 因此客户端不需要使用时要适时取消注册.
Intent metaDataService = new Intent();
metaDataService.setComponent(new ComponentName(MetaDataServiceConfig.META_DATA_SERVICE_PKG_NAME,
MetaDataServiceConfig.META_DATA_SERVICE_CLASS_NAME));
boolean result = bindService(metaDataService, mServiceConnection, Service.BIND_AUTO_CREATE);
Log.d(TAG, "result = " + result); private ServiceConnection mServiceConnection = new ServiceConnection() { @Override
public void onServiceDisconnected(ComponentName componentName) {
try {
if (mIDVRRoadMetaDataTaskBinder != null) {
mIDVRRoadMetaDataTaskBinder.unregisterCallback(0, mCallback);
mIDVRRoadMetaDataTaskBinder = null;
}
} catch (RemoteException e) {
Log.e(TAG, "", e);
}
} @Override
public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
Log.d(TAG, "(DvrMetaDataService - onServiceConnected");
mIDVRRoadMetaDataTaskBinder = IDVRRoadMetaDataTaskBinder.Stub.asInterface(iBinder);
try {
mIDVRRoadMetaDataTaskBinder.registerCallback(0, mCallback);
} catch (RemoteException e) {
Log.e(TAG, "", e);
}
}
}; private IDVRRoadMetaDataTaskBinder mIDVRRoadMetaDataTaskBinder; private IDVRRoadMetaDataCallback mCallback = new IDVRRoadMetaDataCallback.Stub() {
@Override
public void onMetaData(int cameraId, List<RoadLine> roadLines, List<CarData> carDatas) throws RemoteException {
Log.d(TAG, "-> onMetaData :" + carDatas);
// do something needed
}
};
Aidl实现进程间通信,跨进程回调的更多相关文章
- Binder的使用(跨进程——AIDL,非跨进程)
一.Binder类 1.作用:Binder是客户端与服务器端的通信的媒介(连接各种Manager的桥梁),客户端通过Binder对象获取服务器端提供的数据 (为什么要用Binder来提供数据呢,服务器 ...
- Android IPC机制(三)使用AIDL实现跨进程方法调用
上一篇文章中我们介绍了使用Messenger来进行进程间通信的方法,但是我们能发现Messenger是以串行的方式来处理客户端发来的信息,如果有大量的消息发到服务端,服务端仍然一个一个的处理再响应客户 ...
- Android IPC机制(三)在Android Studio中使用AIDL实现跨进程方法调用
在上一篇文章Android IPC机制(二)用Messenger进行进程间通信中我们介绍了使用Messenger来进行进程间通信的方法.可是我们能发现Messenger是以串行的方式来处理client ...
- 【朝花夕拾】跨进程通信,你只知道AIDL,就OUT了
一.前言 提起跨进程通信,大多数人首先会想到AIDL.我们知道,用AIDL来实现跨进程通信,需要在客户端和服务端都添加上aidl文件,并在服务端的Service中实现aidl对应的接口.如果还需要服务 ...
- AIDL跨进程通信
Android跨进程通信会用到AIDL,当然跨进程通信不一定要用AIDL,像广播也是可以的,当然这里用到AIDL相对比较安全一些: AIDL允许传递基本数据类型(Java 的原生类型如int/long ...
- Android AIDL Service 跨进程传递复杂数据
黑夜 黑夜给了我黑色的眼睛,我却用它寻找光明~ 传值方式 AIDL是同意跨进程传递值的,一般来说有三种方式: - 广播:这样的算是比較常见的一种方式了,传递小数据不错 - 文件:这个是保存到文件里.然 ...
- android 远程Service以及AIDL的跨进程通信
在Android中,Service是运行在主线程中的,如果在Service中处理一些耗时的操作,就会导致程序出现ANR. 但如果将本地的Service转换成一个远程的Service,就不会出现这样的问 ...
- Android中的跨进程通信方法实例及特点分析(一):AIDL Service
转载请注明出处:http://blog.csdn.net/bettarwang/article/details/40947481 近期有一个需求就是往程序中增加大数据的採集点,可是由于我们的Andro ...
- android不需要Socket的跨进程推送消息AIDL!
上篇介绍了跨进程实时通讯http://www.cnblogs.com/xiaoxiaing/p/5818161.html 但是他有个缺点就是服务端无法推送消息给客户端,今天这篇文章主要说的就是服务器推 ...
随机推荐
- JS原型链与继承别再被问倒了
原文:详解JS原型链与继承 摘自JavaScript高级程序设计: 继承是OO语言中的一个最为人津津乐道的概念.许多OO语言都支持两种继承方式: 接口继承 和 实现继承 .接口继承只继承方法签名,而实 ...
- leetcode-前K个高频元素
给定一个非空的整数数组,返回其中出现频率前 k 高的元素. 示例 1: 输入: nums = [1,1,1,2,2,3], k = 2 输出: [1,2] 示例 2: 输入: nums = [1], ...
- python常用命令—windows终端查看安装包信息
1, pip list 会将 Python 的所有安装包全部显示出来, 左边是包名, 右边是包的版本号. 2, pip show 包的名字 会将这个包的名字,版本号,包的功能说明,按装这个包的路径显示 ...
- maven项目中没有resource文件夹的问题
之前使用eclipse创建maven项目,文件夹都是建好的,这几次创建,都没有resource文件夹,需要手动创建resource. 现象描述 在eclipse中,创建maven项目有两种方式: 一种 ...
- 【RL系列】Multi-Armed Bandit笔记补充(一)
在此之前,请先阅读上一篇文章:[RL系列]Multi-Armed Bandit笔记 本篇的主题就如标题所示,只是上一篇文章的补充,主要关注两道来自于Reinforcement Learning: An ...
- 由一个hash字符串生成多个子hash字符串
通过存储一个head hash,然后把子hash放到网络中 当然,也可以像默克尔树那样的,生成多级的子hash ,可以通过规则配置不同的hash 生成方式.倒置的默克尔树 我有一个文件,然后我把她分隔 ...
- IntelliJ IDEA for MAC 注释模板、快捷键生成注释
增加注释 在IntelliJ IDEA中为JAVA代码增加注释,首先需要配置注释模板,而后使用模板快捷键生成注释, 下面按照[配置模板].[模板使用]两部分进行介绍 ----------------- ...
- mouseover 和 mouseout 事件是可以冒泡的 取消
mouseover 和 mouseout 事件是可以冒泡的,子元素上触发的事件会冒泡到父元素上.可以改用 mouseleave 和 mouseenter 事件,这两个事件不冒泡.
- c# 委托初窥
1.委托可以把方法当作参数在另一个方法中传递和调用 ,委托是方法的快捷方式. 2.委托是一个类. private void BeginSocketThread() { try { IPEndPoint ...
- java---迭代器(Iterator)
迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构.迭代器通常被称为“轻量级”对象,因为创建它的代价小. Java中的Iterator功能比较简单, ...