Android - base - ContentProvider

四大组件之内容提供器
提供一个外部访问接口,让其他APP访问到我们这个APP的数据。

大纲

  • 内容提供器简介
  • 核心类
  • 内容URI
  • 访问其他APP中的数据
  • 提供自己的内容提供器

#内容提供器简介

内容提供器 Content Provider 主要用于在不同的APP之间实现数据共享。


#核心类

ContentResolver 类:访问其他APP数据的工具类,内置CRUD方法。

获取:Context 类的 getContentResolver() 方法。


#内容URI

内容URI给内容提供器中的数据建立唯一的标识符,同时也是ContentResolver类进行CURD操作的所需参数之一。

内容URI由两部分组成:

  • 权限<authority>:用于对不同的应用程序进行区分。一般情况下,都会采用程序的包名进行区分
  • 路径<path>:用于对同一应用程序中不同的表做区分,添加在去权限的后面。一般情况下,使用表名

再在字符串的头部加上协议声明,故内容URI的标准格式如下:
content:/com.example.cheng.android_contentprovider/book
我们可以在内容URI的后面加上一个id:
content://com.example.cheng.android_contentprovider/book/1
表示调用方期望访问的是 com.example.cheng.android_contentprovider 这个应用的
book 表中 id 为 1 的数据。

两种格式的内容 URI:

  • 以路径结尾就表示期望访问该表中所有的数据
  • 以id结尾表示期望访问该表中拥有响应id的数据

可以使用通配符的方式来分别匹配这两种格式的内容 URI:

  • "*" :表示匹配任意长度的任意字符
  • "#":表示匹配任意长度的数字

例:

一个能匹配任意表的内容 URI 格式就可以写成:
content://com.example.cheng.android_contentprovider/*
一个能匹配 book 表中任意一行数据的内容 URI 就可以写成:
content://com.example.cheng.android_contentprovider/book/#

得到了内容URI字符串,我们需要将之解析成Uri对象才可以做为参数传入:

Uri uri = Uri.parse("content://com.example.app.provider/table1");

访问其他APP中的数据

获取联系人数据 需要运行时权限处理

if (ContextCompat.checkSelfPermission(this, READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) {
    ActivityCompat.requestPermissions(this, new String[] {READ_CONTACTS}, READ_CONTACTS_REQUEST);
} else {
    Cursor result = getContentResolver().query(
            ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
            null,
            null,
            null,
            null);

    if (result != null) {
        while(result.moveToNext()) {
            String name = result.getString(result.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
            String phone = result.getString(result.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
            
            Toast.makeText(this, name + ":" + phone, Toast.LENGTH_SHORT).show();
        }
    }
}

和SQLite操作很相似,但访问外部数据传入的不是表名而是 URL ,query() 方法参数也少了几个。

query

/*
 * @param uri 内容uri "content://com.example.cheng.android_contentprovider/book"
 * @param projection 查询的列
 * @param selection 条件
 * @param selectionArgs 占位值
 * @param sortOrder 排序
 */
public final Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)

其他三个方法的用法类似,不作赘述。

  • 访问外部资源前提是得知道外部资源的内容URI

#提供自己的内容提供器

  1. 创建类继承 ContentProvider 类并实现6个方法。
  2. 在 onCreate() 方法内完成初始化操作。
  3. 将提供给外部访问的 URI 封装到 UriMatcher 对象
// 标识
public static final int BOOK_DIR = 0;
public static final int BOOK_ITEM = 1;
public static final int CATEGORY_DIR = 2;
public static final int CATEGORY_ITEM = 3;

// 包名
public static final String authority = "com.example.cheng.android_contentprovider";

private static UriMatcher uriMatcher;

static {
    uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
    uriMatcher.addURI(authority, "book", BOOK_DIR);
    uriMatcher.addURI(authority, "book/#", BOOK_ITEM);
    uriMatcher.addURI(authority, "category", CATEGORY_DIR);
    uriMatcher.addURI(authority, "category/#", CATEGORY_ITEM);
}

这些 URI 只是给其他APP访问的,并不能直接访问数据安全。我们的内容提供器的CRUD方法被调用时会经过 uriMatcher.match(uri) 方法得到 封装进去的标记,然后我们根据标记写逻辑。

  1. 在 getType() 方法内将 URI 转成 MIME 类型
/*
 * 转成URI对应的MIME类型
 * @param uri
 * @return
 */
@Nullable
@Override
public String getType(@NonNull Uri uri) {
    switch (uriMatcher.match(uri)) {
        case BOOK_DIR:
            return "vnd.android.cursor.dir/vnd.com.example.cheng.android_contentprovider.book";
        case BOOK_ITEM:
            return "vnd.android.cursor.item/vnd.com.example.cheng.android_contentprovider.book";
        case CATEGORY_DIR:
            return "vnd.android.cursor.dir/vnd.com.example.cheng.android_contentprovider.category";
        case CATEGORY_ITEM:
            return "vnd.android.cursor.item/vnd.com.example.cheng.android_contentprovider.category";
        }
        return null;
    }

内容 URI 转 MIME 规则

  • 必须以 vnd. 开头。
  • 如果内容 URI 以路径结尾,则后接 android.cursor.dir/,如果内容 URI 以
  • 结尾,则后接 android.cursor.item/。
  • 最后接上 vnd.<authority>.<path>。

内容URI:content://com.example.cheng.android_contentprovider/book
MIME:android.cursor.dir/vnd.com.example.cheng.android_contentprovider.book
vnd. 开头
android.cursor.dir/ 路径结尾
vnd.com.example.cheng.android_contentprovider.book vnd.<authority>.<path>

query

// uri.getPathSegments().get(1):将路径 "/" 前后切割 1索引是id值
@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String order) {
    Cursor result = null;
    
    // 根据标记得知用户想操作哪张表。
    switch (uriMatcher.match(uri)) {
        case BOOK_DIR:
            result = database.query("book", projection, selection, selectionArgs, null, null, order);
            break;
        case BOOK_ITEM:
            String bid = uri.getPathSegments().get(1);
            result = database.query("book", projection, "id = ?", new String[] {bid}, null, null, order);
            break;
        case CATEGORY_DIR:
            result = database.query("category", projection, selection, selectionArgs, null, null, order);
            break;
        case CATEGORY_ITEM:
            String cid = uri.getPathSegments().get(1);
            result = database.query("category", projection, "id = ?", new String[] {cid}, null, null, order);
            break;
            default:
        }
        return result;
    }

方法内是数据库操作,但是我们不能让外部程序 乱搞 套了一层 让外部程序只能访问到我们想提供的数据 数据安全

CRUD剩余方法

/*
 * @param uri 内容uri
 * @param contentValues 存值对象
 * @return 插入数据的 内容uri   "content://com.example.cheng.android_contentprovider/1"
 */
public Uri insert(@NonNull Uri uri, @Nullable ContentValues contentValues)

/*
 * @param uri 内容uri
 * @param selection 条件
 * @param selectionArgs 占位值
 * @return 影响数据行数
 */
public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs)


/*
 * @param uri 内容uri
 * @param contentValues 存值对象
 * @param selection 条件
 * @param selectionArgs 占位值
 * @return 影响数据行数
 */
public int update(@NonNull Uri uri, @Nullable ContentValues contentValues, @Nullable String selection, @Nullable String[] selectionArgs)
  • 最后别忘了注册内容提供器

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,198评论 6 514
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,334评论 3 398
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,643评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,495评论 1 296
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,502评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,156评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,743评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,659评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,200评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,282评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,424评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,107评论 5 349
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,789评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,264评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,390评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,798评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,435评论 2 359

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,693评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,262评论 25 707
  • 8月14日 星期一 晴 简单的事情重复做,不断的总结优化,整理出适合自己的一套做事方法,才可以不断地进步成长...
    Amy莲子阅读 185评论 0 0
  • 攒够了失望就离开🥀 若有一天你发现 我 慢慢的开始不在乎了[伤心] 慢慢的不再有耐心 慢慢的不...
    梓璐吖阅读 325评论 11 0
  • 古魂闻说重返阳, 魂飘三尺圆梦心。 是道忧愁尽相思, 落阳魂魄欲守知。
    月日明阅读 116评论 0 0