Android中使用ContentProvider进行跨进程方法调用

时间:2022-11-10 08:50:05

原文同一时候发表在我的博客

点我进入还能看到很多其它

需求背景

近期接到这样一个需求,须要和别的 App 进行联动交互,比方下载器 App 和桌面 App 进行联动。桌面的 App 能直接显示下载器 App 内的下载任务进度和状态。

寻找解决方式

从需求上知道了,主要问题在怎样解决跨进程的通信上边。

  1. AIDL

    AIDL 即 Android Interface Definition Language的缩写,是专为 Android 中跨进程通信接口的描写叙述语言。优缺点非常明显,长处是稳定,快。Android 专门用于跨进程通信设计的。缺点是比較麻烦,AIDL 是通信的约定,參加通信的两方都须要把这个 AIDL 文件都加入自己的代码中,然后创建 Service 来实现訪问和被訪问。

  2. ContentProvider

    作为 Android 四大基础组件之中的一个的 ContentProvider 本来它的作用仅仅是提供内容性质的跨进程訪问。可是在 API 11 (Android 3.0) 中,ContentProvider 加入了一个新的方法,能够用来进行跨进程的方法调用,ContentProvider 中这种方法的定义例如以下:

    Bundle call(String method, String arg, Bundle extras)

    从易用性来讲,这个没有 AIDL 那么麻烦,并且扩展性更强,也没有 Broadcast 过于依赖系统,API 11 应该就是主要是缺点了,别的缺点临时没发现。欢迎补充。

  3. Broadcast

    广播是最简单的:长处是把分发消息的任务所有交给 Android 系统了;缺点也是由于全交给系统了,非常多地方不受控制。缺点:

    1. 尽管广播能够通过指定包名来进行发送指向性消息,可是却不能验证消息去向 App 的签名。
    2. 系统重新启动之后,在系统的广播队列里边的消息就丢失了。

实现

为了简要,主要讲讲 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 boolean onCreate() {
return false;
} @Override
public Cursor query(Uri uri, String[] strings, String s, String[] strings2, String s2) {
return null;
} @Override
public String getType(Uri uri) {
return null;
} @Override
public Uri insert(Uri uri, ContentValues contentValues) {
return null;
} @Override
public int delete(Uri uri, String s, String[] strings) {
return 0;
} @Override
public int update(Uri uri, ContentValues contentValues, String s, String[] strings) {
return 0;
} @Override
public Bundle call(String method, String arg, Bundle extras) {
if ("DOWNLOAD".equals(method)) { // 当调用我下载的时候
Log.e("Downloader", "download: " + arg);
// 调用桌面 App 的方法来更新状态
updateStatus("download");
} else ("PAUSE".equals(method)) { // 当调用我暂停的时候
Log.e("Downloader", "pause: " + arg);
// 调用桌面 App 的方法来更新状态
updateStatus("pause");
}
return null;
} // 我们要调用的对方的 ContentProvider 的 URI
private final Uri LAUNCHERCONTENTPROVIDER_URI = Uri.parse("content://cn.hiroz.launcher.LauncherContentProvider");
} private void updateStatus(String status) {
getContext().getContentResolver().call(LAUNCHERCONTENTPROVIDER_URI, "UPDATE_STATUS", status, new Bundle());
}

在下载器 App 的 AndroidManifest.xml 中还须要加入 ContentProvider 的定义:

<provider
android:name="cn.hiroz.downloader.realname.DownloaderContentProvider"
android:authorities="cn.hiroz.downloader.DownloaderContentProvider"
android:exported="true"/>

我特地加了authorities设置。这样在交互时候訪问的 ContentProvider 的 URI 会看起来不一样。也不会暴露我真实的 ContentProvider 类

然后是桌面 App 的 ContentProvider 代码实现

package cn.hiroz.launcher.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 LauncherContentProvider extends ContentProvider {
@Override
public boolean onCreate() {
return false;
} @Override
public Cursor query(Uri uri, String[] strings, String s, String[] strings2, String s2) {
return null;
} @Override
public String getType(Uri uri) {
return null;
} @Override
public Uri insert(Uri uri, ContentValues contentValues) {
return null;
} @Override
public int delete(Uri uri, String s, String[] strings) {
return 0;
} @Override
public int update(Uri uri, ContentValues contentValues, String s, String[] strings) {
return 0;
} @Override
public Bundle call(String method, String arg, Bundle extras) {
// 当被调用“更新状态”的时候
if ("UPDATE_STATUS".equals(method)) {
Log.e("Launcher", "update status: " + arg);
}
return null;
} // 我们要调用的对方的 ContentProvider 的 URI
private final Uri DOWNLOADERCONTENTPROVIDER_URI = Uri.parse("content://cn.hiroz.downloader.DownloaderContentProvider");
} public void download(String arg) {
getContext().getContentResolver().call(DOWNLOADERCONTENTPROVIDER_URI, "DOWNLOAD", status, new Bundle());
} public void pause(String arg) {
getContext().getContentResolver().call(DOWNLOADERCONTENTPROVIDER_URI, "PAUSE", status, new Bundle());
} }

在桌面 App 的 AndroidManifest.xml 中还须要加入 ContentProvider 的定义:

<provider
android:name="cn.hiroz.launcher.realname.LauncherContentProvider"
android:authorities="cn.hiroz.launcher.LauncherContentProvider"
android:exported="true"/>

然后在桌面 App 中。就能够通过 LauncherContentProvider 的 download 方法和 pause 方法来调用下载器 App 的功能了(这两个方法写在这里不太合适,只是我仅仅是为了节省篇幅放一起了)。下载器 App 中被调用了方法,就会调用桌面 App 的更新状态。

这里仅仅是演示了一个交互的过程,有很多其它问题欢迎大家一起讨论学习~~

引申

  • 找不到 ContentProvider 的时候须要做一下空指针保护

  • 签名校验

Android中使用ContentProvider进行跨进程方法调用的更多相关文章

  1. Android IPC机制(三)在Android Studio中使用AIDL实现跨进程方法调用

    在上一篇文章Android IPC机制(二)用Messenger进行进程间通信中我们介绍了使用Messenger来进行进程间通信的方法.可是我们能发现Messenger是以串行的方式来处理client ...

  2. Android IPC机制(三)使用AIDL实现跨进程方法调用

    上一篇文章中我们介绍了使用Messenger来进行进程间通信的方法,但是我们能发现Messenger是以串行的方式来处理客户端发来的信息,如果有大量的消息发到服务端,服务端仍然一个一个的处理再响应客户 ...

  3. Android四大组件应用系列——使用ContentProvider实现跨进程通讯

    一.问题描述 如何在Android中实现不同应用之间的通讯(既跨进程进行调用)?Android提供了多种实现方式,使我们可以实现跨进程访问Activity.通过ContentProvider跨进程访问 ...

  4. Wayland中的跨进程过程调用浅析

    原文地址:http://blog.csdn.net/jinzhuojun/article/details/40264449 Wayland协议主要提供了Client端应用与Server端Composi ...

  5. Android中View的绘制过程 onMeasure方法简述 附有自定义View例子

    Android中View的绘制过程 onMeasure方法简述 附有自定义View例子 Android中View的绘制过程 当Activity获得焦点时,它将被要求绘制自己的布局,Android fr ...

  6. 【转】Android中引入第三方Jar包的方法&lpar;java&period;lang&period;NoClassDefFoundError解决办法&rpar;

    原文网址:http://www.blogjava.net/anchor110/articles/355699.html 1.在工程下新建lib文件夹,将需要的第三方包拷贝进来.2.将引用的第三方包,添 ...

  7. Android中解析XML格式数据的方法

    XML介绍:Extensible Markup Language,即可扩展标记语言 一.概述 Android中解析XML格式数据大致有三种方法: SAX DOM PULL 二.详解 2.1 SAX S ...

  8. Android 中对于图片的内存优化方法

    Android 中对于图片的内存优化方法,需要的朋友可以参考一下     1. 对图片本身进行操作 尽量不要使用 setImageBitmap.setImageResource. BitmapFact ...

  9. android不需要Socket的跨进程推送消息AIDL!

    上篇介绍了跨进程实时通讯http://www.cnblogs.com/xiaoxiaing/p/5818161.html 但是他有个缺点就是服务端无法推送消息给客户端,今天这篇文章主要说的就是服务器推 ...

随机推荐

  1. Css--深入学习之切角

    本文是作者从别的网站和文章学习了解的知识,简单做了个笔记,想要学习更多的可以参考这里:[css进阶]伪元素的妙用--单标签之美,奇思妙想 带切角的矩形: 该图来源于(奇思妙想) Css代码: .not ...

  2. Linux curl使用简单介绍

    在两台新搬迁的微信服务器上执行命令: curl -H "Content-Type: application/json" -d '{"partner_no":&q ...

  3. VPN服务器的配置与应用

      实验场景 通过将Linux配置VPN服务器允许远程计算机能够访问内网. 我的目的: 现在需要开发第三方接口,而第三方接口有服务器IP地址鉴权配置,这样在本地开发出来的程序每次都要发布到服务器上测试 ...

  4. 如何在已安装vs2010的基础上安装sql2008

    以前老受到别人写的这类东西的帮助,所以这次决定自己试下,第一次发这种,写得不好莫怪.       涉略sql2008一个多星期了.怎么说呢?Transact-SQL的编程虽然不如C++,java等高级 ...

  5. Linux Epoll介绍和程序实例

    Linux Epoll介绍和程序实例 1. Epoll是何方神圣? Epoll但是当前在Linux下开发大规模并发网络程序的热门人选,Epoll 在Linux2.6内核中正式引入,和select类似, ...

  6. 使用composer更新thinkphp5或则yii2的版本

    更新thinkphp5或则yii2的版本,我目前采用的是用composer去更新,小伙伴们如果有其他更好的办法更新,可以直接评论给我,不胜感激啊. 如果还没有安装 Composer ,你可以按 Com ...

  7. MySQL系列教程(五)

    MyCAT MyCat是基于阿里开源的Cobar产品而研发,Cobar的稳定性.可靠性.优秀的架构和性能以及众多成熟的使用案例使得MYCAT一开始就拥有一个很好的起点,站在巨人的肩膀上,我们能看到更远 ...

  8. URL和URL比较

    浅谈URI和URL URI(Uniform Resource Identifier)字面上的意思是,统一资源标示符 URL(Uniform Resource Locator),统一资源定位符 光从字面 ...

  9. 正则表达式 re&period;findall 用法

    正则 re.findall 的简单用法(返回string中所有与pattern相匹配的全部字串,返回形式为数组)语法: findall(pattern, string, flags=0) import ...

  10. JEECG 3&period;7&period;3 新春版本发布,企业级JAVA快速开发平台

    JEECG 3.7.3新春版本发布 -  微云快速开发平台 导读           ⊙精美Echart报表 ⊙二维码生成功能 ⊙Online接口改造采用JWT机制 ⊙智能菜单搜索 ⊙代码生成器模板优 ...