7.3.1 ContentResolver的基本用法
通过借助ContentResolver类访问内容提供器中共享的数据,可以通过Context中的getContentResolver()方法获取该类实例。
ContentResolver中的增删改查方法都不接收表名参数,而是使用一个Uri参数代替,这个参数被称为内容URI。内容URI给内容提供器中的数据建立了唯一一个标识符,主要由两部分组成:authority和path。authority是用于对不同的应用程序作区分的,采用程序包和路径的命名方式进行命名。比如某个程序包的名是
com.example.app
那么该程序对应的authority就可以命名为
com.example.app.provider
path则是用于对同一应用程序中不同的表作区分的,通常都会添加到authority的后面。比如某个程序的数据库里存在两张表:table1和table2,这是就可以将path分别命名为/table1和/table2,然后把authority和path进行组合,内容URI就变成了
com.example.app.provider/table1
和
com.example.app.provider/table2
不过,还很难辨认出这两个字符串就是两个内容URI,我们还需要在字符串的头部加上协议声明。因此,内容URI最标准的格式写法如下:
content://com.example.app.provider/table1
content://com.example.app.provider/table2
之后我们还需要将它解析成Uri对象才可以作为参数传入:
Uri uri = Uri.parse("content://com.example.app.provider/table1")
现在可以使用这个Uri对象来查询table1表中的数据了:
Cursor cursor = getContentResolver().query(
uri,
projection,
selection,
selectionArgs,
sortOrder);
查询完成后仍返回一个Cursor对象,这是我们就可以将数据从Cursor对象中逐个读取出来了。读取的思路仍然是通过移动游标的位置来遍历Cursor的所有航,然后再取出每一行中相应列的数据,代码如下:
if (cursor != null) {
while (cursor.moveToNext()) {
String column1 = cursor.getString(cursor.getColumnIndex("column1"));
int column2 = cursor.getInt(cursor.getColumnIndex("column2"));
}
cursor.close();
}
7.4 创建内容提供器的步骤
如果想要实现跨程序贡献数据的功能,官方推荐的方式就是使用内容提供器,可以通过新建一个类去继承ContentProvider的方式来创建一个自己的内容提供器。ContentProvider类中有6个抽象方法,我们在使用子类继承它的时候,需要将6个方法全部重写。
每一个方法都会带有Uri这个参数,这个参数也正是调用ContentResolver的增删改查方法时传递过来的,我们需要对传入的Uri参数进行解析,从中分析出调用方期望访问的表和数据,一个标准内容的URI写法是这样的:
content://com.example.app.provider/table1
这样表示调用方期望访问的是com.example.app这个应用的tale1表中的数据。除此之外,我们还可以在这个内容URI的后面加上一个id,如下所示:
content://com.example.app.provider/table1/1
这就表示调用方期望访问得失com.example.app这个应用的table1表中的id为1的数据。我们可以使用通配符的方式来分别匹配这两种格式的内用URI,规则如下:
- *:表示匹配任意长度的任意字符。
- #:表示匹配任意长度的数字。
所以,一个能够匹配任意表的内容URI格式就可以写成:
content://com.example.app.provider/*
而一个能够匹配table1表中任意一行数据的内容URI格式就可以写成:
content://com.example.app.provider/table1/#
接着,我们在借助UriMatcher这个类就可以轻松地实现匹配内容URI的功能。UriMatcher中提供了一个addURI()方法,这个方法接收3个参数,可以分别吧authority、path和一个自定义代码转进去。这样,当调用UriMatcher的match()方法时,就可以将一个Uri对象传入,返回值是某个能够匹配这个Uri对象所对应的自定义代码,利用这个代码,我们可以判断出调用方期望访问的事那张表中的数据了。
getType()方法是所有内容提供器都必须提供的一个方法,用于获取Uri对象所对应的MIME类型。一个融融URI多对应的MIME字符串主要由3个部分组成,Android对这3个部分做了如下格式规定。
- 必须以vnd开头。
- 如果内容URI以路径结尾,则后接
android.cursor.dir/
,如果内容URI以id结尾,则后接android.cursor.item/。 - 最后接上
vnd.<authority>.<path>
。
所以,对于content://com.example.app.provider/table1
这个内容URI,它所对应的MIME类型就可以写成:
vnd.android.cursor.dir/vnd.com.example.app.provider.table1
对于content://com.example.app.provider.table1/1
这个内容URI,它所对应的MIME类型就可以写成:
vnd.android.cursor.item/vnd.com.example.app.provider.table1