内容提供者ContentProvider与前文的界面Activity、服务Service、广播接收者BroadcastReveiver,并列称为Android的四大组件,均是需要自定义子类继承上述组件类,并在清单文件中静态注册或逻辑代码中动态注册才能正常使用。

android.content.ContentProvider内容提供者类,是用来对其他应用程序提供分享数据内容的组件类,在应用程序间的文件共享一文中,针对Android7.0以上版本所注册的FileProvider文件提供者,便是已经定义过的ContentProvider的使用案例。

分享数据创建

在创建ContentProvider内容提供者之前,首先要保证有要分享的数据存在。这些数据可能来自用户交互,也可能来自远程服务器返回结果。应用程序收到一系列数据后,可以根据不同数据结构,采用前文不同的数据本地保存策略,包括普通键值对形式的SharedPreferences、对象结构的SQLiteDatabase、或者普通二进制的File

之后便可以创建继承自android.content.ContentProvider的自定义内容提供者类,并在其实现的相关方法中完成对本地保存数据的读写。而其他应用程序可以通过访问内容提供者的相关方法,进而读写该应用程序要分享的数据。

分享路径授权

由于内容提供者ContentProvider与其他应用类交互是借助路径定位符Uri类,所以在创建的内容提供者ContentProvider中,要先定义并授权允许该类访问的相关Uri

在创建的内容提供者ContentProvider中需要借助android.content.UriMatcher路径定位匹配类,可将传入的Uri类型转换为普通的int类型,以方便在内容提供者内部匹配区分。

通常在内容提供者内部定义为静态全局变量类型的UriMatcher对象,其只有一参的构造方法UriMatcher(int code),参数值为UriMatcher.NO_MATCH=-1

之后可以调用静态对象的addURI(String authority, String path, int code)方法,增加一对路径定位符与int值的匹配。其中参数 authority 是一串固定的授权字符串,在注册该内容提供者ContentProvider时也必须用到改参数值。参数 path 是追加在 authority 参数之后的子路径,以此区分不同的匹配路径,最终生成的一条数据Uri格式为content://authority/path/id。参数 code 是绑定的int值,与每一种Uri格式为content://authority/path的路径Uri相对应。

在需要根据路径定位符Uri获取int值的地方,通常是下文分享数据操作中,可以调用静态对象的match(Uri uri)方法,如果匹配到的Uri与上文authority+path组合一致时,返回对应code值,否则返回值为UriMatcher.NO_MATCH=-1

分享数据操作

ContentProvider中必须实现以下方法以操作本地数据。

创建onCreate()

与其他组件类似,只有在该内容提供者被系统首次加载创建时,系统才会调用该方法。可以在该方法中初始化相关全局变量,但是不建议执行耗时操作。

获取数据类型getType(Uri uri)

由使用该内容提供者的其他应用程序调用。传入参数 uri 作为路径定位符指定一条数据位置,返回该数据对应的MIME类型。通常在 uri 指向文件时,根据不同文件类型以返回对应的MIME类型。其他情况返回null即可。

增加数据insert(Uri uri, ContentValues values)

由使用该内容提供者的其他应用程序调用。在指定的参数 uri 指定的路径下插入一条数据,其内容格式为content://authority/path。参数 values 则是要插入的数据内容,其与数据库中的插入方法类似,以key-value键值对形式保存数据。最终在数据插入成功后,需要返回该条数据对应的路径定位符Uri对象。

删除数据delete(Uri uri, String selection, String[] selectionArgs)

由使用该内容提供者的其他应用程序调用。参数 uri 指定要删除数据所在位置,当内容格式为content://authority/path/id类型时,将删除其指定位置数据,余下两个参数无效;当内容格式为content://authority/path类型时,将删除该路径下根据其他参数查找匹配的数据,若余下参数为null,则删除该路径下所有数据。参数 selection 为指定删除条件,其符合sql语句,但变量参数可用符合?代替,在后边参数中指定具体参数值;参数 selectionArgs 即为参数值数组,长度与参数 selection 中的?符合数量一致。最终返回int类型的数据标记总共删除的数据条数。

修改更新数据update(Uri uri, ContentValues values, String selection, String[] selectionArgs)

由使用该内容提供者的其他应用程序调用。参数 uri 指定要修改数据所在位置,与上文删除数据方法中的参数一类似,当内容格式为content://authority/path/id类型时,将更新其指定位置数据,最后两个参数无效;当内容格式为content://authority/path类型时,将更新该路径下根据最后两个参数查找匹配的数据,若最后两个参数为null,则更新该路径下所有数据。参数 values 是要插入的数据内容,与增加数据方法中的参数二类似。参数 selection 和参数 selectionArgs 同样与删除数据方法汇总的参数二和参数三类似,可以匹配符合要求的数据。最终返回int类型的数据标记总共修改的数据条数。

查询数据query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)系列方法

由使用该内容提供者的其他应用程序调用。参数 uri 为指定查询路径,与增加数据方法中的参数一类似,格式为content://authority/path。余下参数与数据库中的查询方法中的参数类似,均可根据相关内容匹配数据。最终返回android.database.Cursor游标指针类型的数据,同样可参考数据库中的查询结果使用。

清单文件注册

最终自定义的内容提供者需要在清单文件中静态注册,才能在其他位置使用。在<application></application>标签中使用<provider></provider>标签指定内容提供者。在该标签中,android:name属性绑定代码中定义的内容提供者全局类名。属性android:authorities指定授权名称,与代码中UriMatcher对象的添加路径匹配符方法所传入的 authority 参数一致。属性android:exported通过设置boolean类型值,以决定该内容提供者是否可对其他应用程序提供使用。除此之外还有其他一些属性值,可参考官方<provider>介绍。

借助路径定位符访问分享数据

在其他应用程序或当前应用程序的其他位置,都可借助上下文环境Context对象的getContentResolver()方法,获得android.content.ContentResolver内容解析类的对象。通过对内容解析类对象的操作,进而影响自定义的内容提供者。内容解析类ContentResolver中有一系列方法,均与上述实现的自定义内容提供者ContentProvider中的增删改查等系列方法相对应,便不再赘述。


通常定义内容提供者的定义场景,是在短信类、通话类应用中将消息或联系人信息提供给其他应用程序使用。而数据访问方只需要获取对应的Uri即可。

Android系统编程入门系列之应用间数据共享ContentProvider的更多相关文章

  1. Android系统编程入门系列之加载界面Activity

    上回说到应用初始化加载及其生命周期,在Android系统调用Applicaiton.onCreate()之后,继续创建并加载清单文件中注册的首个界面即主Activity,也可称之为入口界面.主Acti ...

  2. Android系统编程入门系列之应用级文件在应用程序间的共享

    在上篇文章了解到应用级文件只能被其所创建的应用程序所访问,那么其他应用程序是不是就无论如何都无法访问了呢?肯定不是的,只要文件经过其创建的应用程序授权,还是可以被其他应用程序所访问的.这也就是应用级文 ...

  3. Android系统编程入门系列之应用环境及开发环境介绍

        作为移动端操作系统,目前最新的Android 11.0已经发展的比较完善了,现在也到了系统的整理一番的时间,接下来的系列文章将以Android开发者为中心,争取用归纳总结的态度对初级入门者所应 ...

  4. Android系统编程入门系列之界面Activity绘制展示

    上篇文章介绍了界面Activity的启动方式和生命周期,本篇将继续介绍在界面Activity中的内容是如何绘制展示给用户的. 在Android系统上运行新创建的界面Activtiy,给用户展示的是空白 ...

  5. Android系统编程入门系列之界面Activity交互响应

    在上篇文章中已经了解到界面Activity的绘制完全依赖其加载的视图组件View,不仅如此,用户的每次触摸操作都可以在界面Activity内接收并响应,也可以直接传递给其中的某个视图View响应.本文 ...

  6. Android系统编程入门系列之界面Activity响应丝滑的传统动画

    上篇文章介绍了应用程序内对用户操作响应的相关方法位置,简单的响应逻辑可以是从一个界面Activity跳转到另一个界面Activity,也可以是某些视图View的相对变化.然而不管是启动一个界面执行新界 ...

  7. Android系统编程入门系列之加载服务Service

    之前几篇文章简单梳理了在Android系统的四大组件之一,最主要的界面Activity中,使应用程序与用户进行交互响应的相关知识点,那对于应用程序中不需要与用户交互的逻辑,又要用到哪些内容呢?本文开始 ...

  8. Android系统编程入门系列之服务Service齐头并进多线程任务

    在上篇文章中初步了解了Android系统的四大组件之一的服务Service,在服务内可以执行无用户交互的耗时操作任务,但是包括之前关于界面系列文章在内,生命周期方法都是在主线程内被系统回调的.如果直接 ...

  9. Android系统编程入门系列之服务Service中的进程间通信

    在上篇文章以线程间的通信方式Handler类结尾,服务Service还支持的进程间通信,又是具体怎么实现的呢?这就要用到加载服务一文中提到的AIDL语言规范了. AIDL是 Android Inter ...

随机推荐

  1. sychronized

    说一下 synchronized 底层实现原理? synchronized可以保证方法或者代码块在运行时, 同一时刻只有一个方法可以进入到临界区, 同时它还可以保证共享变量的内存可见性. Java中每 ...

  2. centos7 误用 cat 打开了一个很大的文件

    2021-09-01 1. 问题描述 刚才看到一个文件,出于好奇我就直接用 cat 命令查看了一下,结果文件巨大,一直刷屏停不下来 2. 解决方法 克隆一个窗口,抓一下这个 cat 进程,再使用 ki ...

  3. Netty ServerBootstrap如何绑定端口

    这篇讲netty服务端ServerBootstrap如何启动 前言 BootStrap在netty的应用程序中负责引导服务器和客户端.netty包含了两种不同类型的引导: 使用服务器的ServerBo ...

  4. Django——实现评论功能(包括评论回复)

    提示:(1)功能不全面,仅仅实现评论(2)样式简单 1.项目目录结构 2.模型 from django.db import models from django.contrib.auth.models ...

  5. 20200713晚 noip14

    考场 很紧张,上午考太烂了 开场看到"影魔",想起以前看过(但没做),心态爆炸,咆哮时被 hkh diss 了 T1 一开始想建边跑最长路,每个点在记录一下 \(\min\{a\} ...

  6. Python习题集(五)

    每天一习题,提升Python不是问题!!有更简洁的写法请评论告知我! https://www.cnblogs.com/poloyy/category/1676599.html 题目 打印99乘法表 解 ...

  7. Selenium系列(十七) - Web UI 自动化基础实战(4)

    如果你还想从头学起Selenium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1680176.html 其次,如果你不懂前端基础知识, ...

  8. Java关键字-static汇总

    1.静态变量(被static修饰的成员变量) 1-静态变量 运行时,Java虚拟机只为静态变量分配一次内存(加载类的过程中完成静态变量的内存分配) 类的内部可以在任何方法内直接访问静态变量 其他类中可 ...

  9. Identity用户管理入门一(框架搭建)

    理论知识微软官方文档最完整,最详细,这里只一步步的介绍如何使用,地址:https://docs.microsoft.com/zh-cn/aspnet/core/security/authenticat ...

  10. Java中使用jxl.jar将数据导出为excel文件

      Java对Excel文件的读写操作可由jxl.jar或poi.jar实现,这里使用jxl.jar完成对Excel文件的导出. 一.将Excel文件导出在本地 步骤:   创建文件 -> 创建 ...