原文同时发表在我的博客
点我进入还能看到更多
需求背景
最近接到这样一个需求,需要和别的 App 进行联动交互,比如下载器 App 和桌面 App 进行联动,桌面的 App 能直接显示下载器 App 内的下载任务进度和状态。
寻找解决方案
从需求上知道了,主要问题在如何解决跨进程的通信上边。
AIDL
AIDL 即 Android Interface Definition Language的缩写,是专为 Android 中跨进程通信接口的描述语言。优缺点很明显,优点是稳定,快,Android 专门用于跨进程通信设计的。缺点是比较麻烦,AIDL 是通信的约定,参加通信的双方都需要把这个 AIDL 文件都加入自己的代码中,然后创建 Service 来实现访问和被访问。
ContentProvider
作为 Android 四大基础组件之一的 ContentProvider 本来它的作用只是提供内容性质的跨进程访问。但是在 API 11 (Android 3.0) 中,ContentProvider 加入了一个新的方法,可以用来进行跨进程的方法调用,ContentProvider 中这个方法的定义如下:
Bundle call(String method, String arg, Bundle extras)
从易用性来讲,这个没有 AIDL 那么麻烦,而且扩展性更强,也没有 Broadcast 过于依赖系统,API 11 应该就是主要是缺点了,别的缺点暂时没发现,欢迎补充。
Broadcast
广播是最简单的:优点是把分发消息的任务全部交给 Android 系统了;缺点也是因为全交给系统了,很多地方不受控制。缺点:
- 虽然广播可以通过指定包名来进行发送指向性消息,但是却不能验证消息去向 App 的签名。
- 系统重启之后,在系统的广播队列里边的消息就丢失了。
实现
为了简要,主要讲讲 ContentProvider 吧。
ContentProvider
首先是下载器 App 的 ContentProvider 代码实现
package cn.hiroz.downloader.realname;
import android.content.ContentProvider;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.Binder;
import android.os.Bundle;
import android.util.Log;
public class DownloaderContentProvider extends ContentProvider {
@Override
public