Android ContentProvider

一.ContentProvider 使用场景:

1.ContentProvider为存储和读取数据提供了统一的接口

  1. 使用ContentProvider,应用程序可以实现数据共享

3.android内置的许多数据都是使用ContentProvider形式,供开发者调用的(如视频,音频,图片,通讯录等)

所以ContentProvider很适合跨进程通信,另外ContentProvider的底层是采用 Android中的Binder机制(因此Binder机制很重要)。

发现问题:

此外但是在写ContentProvider的时候,用的它跨进程通信。举个例子:A应用通过ContentProvider暴露有个接口,使B应用可以访问,此时我就考虑到一个问题,如果A应用被系统杀掉,那么B应用还可不可以获取到A应用暴露的数据,那么OK,我们带着问题去写这个ContentProvider。

二.ContentResolver

在ContentProvider的使用过程中,需要借用ContentResolver来控制ContentProvider所暴露处理的接口,作为代理来间接操作ContentProvider以获取数据。
在 Context.java 的源码中如下抽象方法

public abstract ContentResolver getContentResolver();

所以可以在所有继承Context的类中通过 getContentResovler() 方法获取ContentProvider

  ContentResolver contentResolver = getContentResovler();

三.写一个A应用MyContentProvider

1.首先在Manifest文件中注册provider,并且定义provider的uri和name

         <provider android:name="MyProvider"
                   android:authorities="cn.test.myprovider"
                   android:exported="true"
                />

2.自定义一个MyContentProvider需要继承ContentProvider并重写其中的方法。

public class MyProvider extends ContentProvider {

      private Context mContext;
      DBHelper mDbHelper = null;
      SQLiteDatabase db = null;
      public static final String AUTOHORITY = "cn.scu.myprovider";
      // 设置ContentProvider的唯一标识

      public static final int User_Code = 1;
      public static final int Job_Code = 2;

     // UriMatcher类使用:在ContentProvider 中注册URI
     private static final UriMatcher mMatcher;
       static{
            mMatcher = new UriMatcher(UriMatcher.NO_MATCH);
            // 初始化
           mMatcher.addURI(AUTOHORITY,"user", User_Code);
           mMatcher.addURI(AUTOHORITY, "job", Job_Code);
          // 若URI资源路径 = content://cn.scu.myprovider/user ,则返回注册码User_Code
          // 若URI资源路径 = content://cn.scu.myprovider/job ,则返回注册码Job_Code
  }

  // 以下是ContentProvider的6个方法

   /**
    * 初始化ContentProvider
    */
  @Override
   public boolean onCreate() {

         mContext = getContext();
         // 在ContentProvider创建时对数据库进行初始化
         // 运行在主线程,故不能做耗时操作,此处仅作展示
         mDbHelper = new DBHelper(getContext());
         db = mDbHelper.getWritableDatabase();

         // 初始化两个表的数据(先清空两个表,再各加入一个记录)
         //db.execSQL("delete from user");
        //db.execSQL("insert into user values('null','Carson');");
        //db.execSQL("insert into user values('null','Kobe');");
        //db.execSQL("insert into user values(5,'zenmeban');");

        db.execSQL("delete from job");
        db.execSQL("insert into job values(1,'Android');");
        db.execSQL("insert into job values(2,'iOS');");

        return true;
  }

   /**
     * 添加数据
     */

   @Override
   public Uri insert(Uri uri, ContentValues values) {

        // 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
        // 该方法在最下面
        String table = getTableName(uri);
        ContentValues contentValues = new ContentValues();
       //contentValues.put("_id","4");
       //contentValues.put("name","buzhidao");
       // db.execSQL("insert into user values(null,'buzhidao');");
       // 向该表添加数据
        db.insert(table, null, contentValues);

       // 当该URI的ContentProvider数据发生变化时,通知外界(即访问该ContentProvider数据的访问者)
       mContext.getContentResolver().notifyChange(uri, null);

       // 通过ContentUris类从URL中获取ID
      //long personid = ContentUris.parseId(uri);
      //System.out.println(personid);
       Log.i("wangrui",""+contentValues);
       return uri;
    }

   /**
     * 查询数据
     */
   @Override
   public Cursor query(Uri uri, String[] projection, String selection,
                    String[] selectionArgs, String sortOrder) {
        // 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
       // 该方法在最下面
      String table = getTableName(uri);

      // 通过ContentUris类从URL中获取ID
      //long personid = ContentUris.parseId(uri);
      //System.out.println(personid);

    // 查询数据
    return db.query(table,projection,selection,selectionArgs,null,null,sortOrder,null);
  }

  /**
   * 更新数据
   */
  @Override
  public int update(Uri uri, ContentValues values, String selection,
                  String[] selectionArgs) {

      String table = getTableName(uri);
      int count = db.update(table,values,selection,selectionArgs);
      if(count>0) {
          mContext.getContentResolver().notifyChange(uri, null);
      }
      return 0;
  }

   /**
    * 删除数据
    */
   @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {
          // 由于不展示,此处不作展开
         String table = getTableName(uri);

          return 0;
  }

   @Override
    public String getType(Uri uri) {

       // 由于不展示,此处不作展开
       return null;
   }

   /**
     * 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
     */
    private String getTableName(Uri uri){
        String tableName = null;
        switch (mMatcher.match(uri)) {
            case User_Code:
                  tableName = DBHelper.USER_TABLE_NAME;
                  break;
           case Job_Code:
                 tableName = DBHelper.JOB_TABLE_NAME;
                 break;
         }
         return tableName;
     }
}

3.创建数据库DBHelper

public class DBHelper extends SQLiteOpenHelper {

      // 数据库名
      private static final String DATABASE_NAME = "finch.db";

       // 表名
      public static final String USER_TABLE_NAME = "user";
      public static final String JOB_TABLE_NAME = "job";

      private static final int DATABASE_VERSION = 1;
      //数据库版本号

   public DBHelper(Context context) {
        super(context, DATABASE_NAME, null, DATABASE_VERSION);
   }

     @Override
      public void onCreate(SQLiteDatabase db) {

    // 创建两个表格:用户表 和职业表
         db.execSQL("CREATE TABLE IF NOT EXISTS " + USER_TABLE_NAME + "(_id INTEGER PRIMARY KEY AUTOINCREMENT," + " name TEXT)");

         db.execSQL("CREATE TABLE IF NOT EXISTS " + JOB_TABLE_NAME + "(_id INTEGER PRIMARY KEY AUTOINCREMENT," + " job TEXT)");
}

   @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)   {

    }
}

四 .写一个查找这个provider的另一个demo,B应用

这个demo中只是用来查找前一个应用提供的 contentProvider,只贴出关键代码

       cursor = getContentResolver().query(uri_user, null, null, null, null);
           if (cursor != null) {
              while (cursor.moveToNext()) {
              // int displayName = cursor.getInt(0);
              // String number = cursor.getString(1);
              // String time = cursor.getString(2);
             // 获取联系人姓名
              String displayName = cursor.getString(cursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
             //获取联系人姓名
             String number = cursor.getString(cursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.Phone.NUMBER));

              contactsList.add(displayName + "\n" + number + " ..");
          }
        cursor.close();
    }

ok,两个demo到这里就写完了,最后结果说明,从B应用中可以成功获取到A应用的提供的contentProvider,此时如果把A应用从后台杀掉也是可以成功获取到数据。后续会将两个应用的demo上传GitHub。

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

推荐阅读更多精彩内容