简介
ContentProvider 是Android中提供的专门用于不同应用进行数据共享的方式,它是一种进程间的通信,底层是用Binder实现的。ContentProvider是系统为我们封装的,使得我们无须关心底层细节即可轻松实现IPC。
系统给我们提供了许多ContentProvider,比如通信录,日程表信息等,要跨进程访问这些信息,只需要通过ContentResolver的query、update、insert和delete方法即可。自定义一个ContentProvider只需继承ContentProvider类并实现六个抽象方法即可:onCreate、query、update、insert、delete和getType。onCreate代表ContentProvider的创建,一般来说我们需要做一些初始化工作;getType用来返回一个Uri请求所对应的MIME类型(媒体类型),比如图片、视频等,如果我们应用不关注这个选项,可以直接在这个方法中返回null或者“/”。根据Binder的工作原理,我们知道这六个方法均运行在ContentProvider的进程中,除了onCreate由系统回调并运行在主线程中,其他五个方法均由外界回调并运行在Binder线程池中。
ContentProvider对底层的数据存储方式没有任何要求,既可以使用SQLite数据库,也可以使用普通文件,甚至可以采用内存中的一个对象来进行数据存储。
使用
<provider
android:name=".xxx.xxxProvider"
android:authorities="com.cwx.test.xxxProvider"
android:permission="com.cwx.provider"
android:process=":provider"
>
</provider>
android:authorities是ContentProvider的唯一标识,通过这个属性外部应用就可以访问我们的ContentProvider,建议命名时加上包名前缀。android:process声明ContentProvider运行于独立的进程。android:permission声明外界需要访问该ContentProvider时需要声明该权限。如果声明android:readPermission和android:writePermission属性,则外部应用也必须声明相应的权限才可以进行读/写操作,否则会异常终止。
java代码中访问ContentProvider的关键代码如下:
Uri uri = Uri.parse("content://com.cwx.test.xxxProvider");
getContentResolver().query(uri,null,null,null,null);
具体细节
ContentProvider通过Uri来区分外界要访问的数据集合。我们可以用UriMatcher的addURI方法将Uri和Uri_Code 关联起来。当外界访问我们的ContentProvider时,我们可以根据携带过来的Uri来得到Uri_Code,并根据Uri_Code匹配外界要访问的表,然后进行相应的操作。
具体例子如下:
public static final String AUTHORITY = "com.cwx.test.provider";
public static final Uri TEST1_PROVIDER = Uri.parse("content://"+AUTHORITY +"/test1");
public static final Uri TEST2_PROVIDER = Uri.parse("content://"+AUTHORITY +"/test2");
public static final int TEST1_URI_CODE = 0;
public static final int TEST2_URI_CODE =1;
private static final UriMatcher sUriMatcher = new UriMatcher(UriMatcher.No_MATCH);
static{
sUriMatcher.addURI(AUTHORITY,"test1",TEST1_URI_CODE);
sUriMatcher.addURI(AUTHORITY,"test2",TEST2_URI_CODE);
}
通过以上代码设置,外界就可以通过uri访问到ContentProvider具体对应的表了。
string name = "";
switch(sUriMatcher.match(uri)){
case TEST1_URI_CODE:
name = "test1_name";//test1_name为contentProvider维护的一个数据库表
break;
case TEST2_URI_CODE:
name = "test2_name";
break;
}
注意点
- 当通过增删改方法导致ContentProvider数据发生变化时需要通过ContentResolver的notifyChange方法来通知外界数据发生改变。
- 可以调用ContentResolver的registerContentObserver方法来注册观察者,通过unregisterContentObserver方法来反注册观察者。
- query、update、insert、delete存在多线程并发访问,需要做好线程同步。