创建内容提供器
1.继承ContentProvider类
自定义内容提供器可以通过新建一个类去继承
ContentProvider
来创建自己的内容提供器,该类一共有6个抽象方法,需要我们来重写。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38public class MyProvider extends ContentProvider {
// 初始化内容提供器的时候调用,通常在这里完成数据库创建和升级,返回true代表成功
public boolean onCreate() {
return false;
}
// uri:表名 projection:列名 selection(args):约束查询哪些行 sortOrder:排序 查询的结果在Cursor中返回
public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder) {
return null;
}
public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {
return null;
}
public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {
return 0;
}
public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs) {
return 0;
}
// 根据传入的uri返回相应的MIME【后面解释】
public String getType(@NonNull Uri uri) {
return null;
}
}2.URI 参数解析
内容URI的格式主要有两种
- 1.以路径结尾
content://com.example.app.provider/table1
,表示期望访问该表中的所有的数据 - 2.以id结尾
content://com.example.app.provider/table1/1`,表示期望访问该表中拥有相应id的数据
- 1.以路径结尾
我们可以用
通配符
的方式来分别匹配这两种格式的内容URI
,规则如下:- *:表示匹配任意长度的任意字符
- #:表示匹配任意长度的数字
举例:
- 一个能够匹配任意表的内容URI格式可以写成:
content://com.example.app.provider/*
- 一个能够匹配
table
表中任意一行数据的内容URI格式就可以写成:content://com.example.app.provider/table/#
- 一个能够匹配任意表的内容URI格式可以写成:
借助
UriMatcher类
实现匹配内容URI的功能UriMatcher类
提供了一个addURI()
方法,该方法接收3个参数,authority
、path
和自定义代码
当调用
UriMatcher
的match()
方法时,将一个Uri对象
传入,返回值就是某个能够匹配这个Uri对象所对应的自定义代码
,利用这个代码,就可以判断出调用方期望访问的是哪张表中的数据了。
query() 示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37public class MyProvider extends ContentProvider {
//创建4个常量,代表调用方期望访问的不同内容
public static final int TABLE1_DIR = 0; //期望访问table1所有数据
public static final int TABLE1_ITEM = 1; //期望访问table1某一条数据
public static final int TABLE2_DIR = 2; //期望访问table2所有数据
public static final int TABLE2_ITEM = 3; //期望访问table2某一条数据
private static UriMatcher uriMatcher;
static {
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH); //创建UriMatcher实例
uriMatcher.addURI("com.example.app.provider", "tabke1", TABLE1_DIR);
uriMatcher.addURI("com.example.app.provider", "tabke1/#", TABLE1_ITEM);
uriMatcher.addURI("com.example.app.provider", "tabke2", TABLE2_DIR);
uriMatcher.addURI("com.example.app.provider", "tabke2/#", TABLE2_ITEM);
}
// uri:表名 projection:列名 selection(args):约束 sortOrder:排序 查询的结果在Cursor中返回
public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder) {
switch (uriMatcher.match(uri)) {
case TABLE1_DIR:
// 查询表一所有数据
break;
case TABLE1_ITEM:
// 查询表一单条数据
break;
case TABLE2_DIR:
break;
case TABLE2_ITEM:
break;
default:
}
return null;
}
}
getType()
方法getType()
方法,是所有内容提供器都必须提供的一个方法,用于获取Uri对象
所对应的MIME类型
MIME
由三部分组成:- 必须以
vnd
开头 - 如果
URI
以路径结尾,则在vnd
后接android.cursor.dir/
;如果URI
以ID结尾,则后接android.cursor.item/
- 最后接上
vnd.<authority>.<path>
例如:
1
2
3content://com.example.app.provider/table1
则为:
vnd.android.cursor.dir/vnd.com.example.app.provider/table1所以
getType():
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15public String getType(@NonNull Uri uri) {
switch (uriMatcher.match(uri)) {
case TABLE1_DIR:
return "vnd.android.cursor.dir/vnd.com.example.app.provider.table1";
case TABLE1_ITEM:
return "vnd.android.cursor.item/vnd.com.example.app.provider.table1";
case TABLE2_DIR:
return "vnd.android.cursor.dir/vnd.com.example.app.provider.table2";
case TABLE2_ITEM:
return "vnd.android.cursor.item/vnd.com.example.app.provider.table2";
default:
break;
}
return null;
}- 必须以