ContentProvider总结

简介

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存在多线程并发访问,需要做好线程同步。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 139,698评论 20 593
  • 1. 夜闌人靜路燈斜倚著昏黯的剪影 一隻晚睡的彩蝶在枝椏間撲騰 天際的流星寂寞清冷 2. 頑皮的孩子從夢酣处驚醒 ...
    安子觅阅读 81评论 2 14
  • 今年高二了,我发现班上的男生越来越讨厌了,居然给我起了个外号叫女包公!一见到我就大唱“开封有个包青天……” 真是的...
    护肤001阅读 139评论 1 0
  • 用心的准备食材 为你做一餐饭 也许这样的机会 以后不会再有 就算你坐在我的身边 相同的心境又如何能持续 一如你现在...
    子执年华阅读 323评论 0 2