一、ContentProvider简介

当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据。ContentProvider为存储和获取数据提供了统一的接口。虽然使用其他方法也可以对外共享数据,但数据访问方式会因数据存储的方式而不同,如采用文件方式对外共享数据,需要进行文件操作读写数据;采用sharedpreferences共享数据,需要使用sharedpreferences API读写数据。而使用ContentProvider共享数据的好处是统一了数据访问方式。

query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder)
通过Uri进行查询,返回一个Cursor

insert(Uri url, ContentValues values)
将一组数据插入到Uri 指定的地方

update(Uri uri, ContentValues values, String where, String[] selectionArgs)
更新Uri指定位置的数据

delete(Uri url, String where, String[] selectionArgs)
删除指定Uri并且符合一定条件的数据

二、Uri类简介

Uri代表了要操作的数据,Uri主要包含了两部分信息
①需要操作的ContentProvider
②对ContentProvider中的什么数据进行操作

组成部分
①scheme:ContentProvider的scheme已经由Android所规定为content://
②主机名(Authority):用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来找到它。建议为公司域名,保持唯一性
③路径(path):可以用来表示我们要操作的数据,路径的构建应根据业务而定:

要操作person表中id为10的记录
content://cn.xyCompany.providers.personProvider/person/10

要操作person表中id为10的记录的name字段
content://cn.xyCompany.providers.personProvider/person/10/name

要操作person表中的所有记录
content://cn.xyCompany.providers.personProvider/person

要操作的数据不一定来自数据库,也可以是文件等他存储方式,如要操作xml文件中user节点下的name节点

content://cn.xyCompany.providers.personProvider/person/10/name

把一个字符串转换成Uri,可以使用Uri类中的parse()方法
Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person")

三、UriMatcher、ContentUris和ContentResolver简介

Uri代表了要操作的数据,所以经常需要解析Uri,并从Uri中获取数据。Android系统提供了两个用于操作Uri的工具类,分别为UriMatcher 和ContentUris。掌握它们的使用会便于我们的开发工作。

UriMatcher

用于匹配Uri

①把需要匹配Uri路径全部给注册上

// 常量UriMatcher.NO_MATCH表示不匹配任何路径的返回码(-1)。
UriMatcher  uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);

// 若match()方法匹配content://cn.xyCompany.providers.personProvider/person路径则返回匹配码为1
uriMatcher.addURI("content://cn.xyCompany.providers.personProvider","person", 1);

// 若match()方法匹配content://cn.xyCompany.providers.personProvider/person/10路径则返回匹配码为2
uriMatcher.addURI("content://cn.xyCompany.providers.personProvider","person/#", 1);

②注册完需要匹配的Uri后,就可以使用uriMatcher.match(uri)方法对输入的Uri进行匹配

ContentUris
ContentUris是对URI的操作类,其中的withAppendedId(uri, id)用于为路径加上ID部分,parseId(uri)方法用于从路径中获取ID部分方法很实用。
Uri insertUri = Uri.parse("content://cn.xyCompany.providers.personProvider/person" + id);等价于
Uri insertUri = ContentUris.withAppendedId(uri, id);

ContentResolver
当外部应用需要对ContentProvider中的数据进行添加、删除、修改和查询操作时,可以使用ContentResolver 类来完成。要获取ContentResolver 对象,可以使用Activity提供的getContentResolver()方法。 ContentResolver使用insert、delete、update、query方法来操作数据。

三、实例代码

当数据需要在应用程序间共享时,我们就可以利用ContentProvider为数据定义一个URI。之后其他应用程序对数据进行查询或者修改时,只需要从当前上下文对象获得一个ContentResolver(内容解析器)传入相应的URI就可以了。

contentProvider和Activity一样是Android的组件,故使用前需要在AndroidManifest.xml中注册,必须放在主应用所在包或其子包下。

<application android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".MainActivity"
                  android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
            <intent-filter>
                <data android:mimeType="vnd.android.cursor.dir/person" />
            </intent-filter>
            <intent-filter>
                <data android:mimeType="vnd.android.cursor.item/person" />
            </intent-filter>
        </activity>
        <!-- 配置内容提供者,android:authorities为该内容提供者取名作为在本应用中的唯一标识 -->
        <provider android:name=".providers.PersonProvider"
                android:authorities="cn.xyCompany.providers.personProvider"/>
    </application>

内容提供者和测试代码

内容提供者
package cn.xy.cotentProvider.app.providers;
import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.net.Uri;
import android.util.Log;
import cn.xy.cotentProvider.service.DBOpeningHelper;

public class PersonProvider extends ContentProvider
{
 private DBOpeningHelper dbHelper;

// 若不匹配采用UriMatcher.NO_MATCH(-1)返回
 private static final UriMatcher MATCHER = new UriMatcher(UriMatcher.NO_MATCH);

// 匹配码
 private static final int CODE_NOPARAM = 1;
 private static final int CODE_PARAM = 2;

static
 {
  // 对等待匹配的URI进行匹配操作,必须符合cn.xyCompany.providers.personProvider/person格式
  // 匹配返回CODE_NOPARAM,不匹配返回-1
  MATCHER.addURI("cn.xyCompany.providers.personProvider", "person", CODE_NOPARAM);

// #表示数字 cn.xyCompany.providers.personProvider/person/10
  // 匹配返回CODE_PARAM,不匹配返回-1
  MATCHER.addURI("cn.xyCompany.providers.personProvider", "person/#", CODE_PARAM);
 }

@Override
 public boolean onCreate()
 {
  dbHelper = new DBOpeningHelper(this.getContext());
  return true;
 }

@Override
 public Uri insert(Uri uri, ContentValues values)
 {
  SQLiteDatabase db = dbHelper.getWritableDatabase();
  switch (MATCHER.match(uri))
  {
   case CODE_NOPARAM:
    // 若主键值是自增长的id值则返回值为主键值,否则为行号,但行号并不是RecNo列
    long id = db.insert("person", "name", values);
    Uri insertUri = ContentUris.withAppendedId(uri, id);
    return insertUri;
   default:
    throw new IllegalArgumentException("this is unkown uri:" + uri);
  }
 }

@Override
 public int delete(Uri uri, String selection, String[] selectionArgs)
 {
  SQLiteDatabase db = dbHelper.getWritableDatabase();
  switch (MATCHER.match(uri))
  {
   case CODE_NOPARAM:
    return db.delete("person", selection, selectionArgs); // 删除所有记录
   case CODE_PARAM:
    long id = ContentUris.parseId(uri); // 取得跟在URI后面的数字
    Log.i("provider", String.valueOf(id));
    String where = "id = " + id;
    if (null != selection && !"".equals(selection.trim()))
    {
     where += " and " + selection;
    }
    return db.delete("person", where, selectionArgs);
   default:
    throw new IllegalArgumentException("this is unkown uri:" + uri);
  }
 }

@Override
 public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
 {
  SQLiteDatabase db = dbHelper.getWritableDatabase();
  switch (MATCHER.match(uri))
  {
   case CODE_NOPARAM:
    return db.update("person",values,selection, selectionArgs); // 更新所有记录
   case CODE_PARAM:
    long id = ContentUris.parseId(uri); // 取得跟在URI后面的数字
    String where = "id = " + id;
    if (null != selection && !"".equals(selection.trim()))
    {
     where += " and " + selection;
    }
    return db.update("person",values,where,selectionArgs);
   default:
    throw new IllegalArgumentException("this is unkown uri:" + uri);
  }
 }
 
 
 @Override
 public String getType(Uri uri)
 {
  switch(MATCHER.match(uri))
  {
   case CODE_NOPARAM:
    return "vnd.android.cursor.dir/person";
   case CODE_PARAM:
    return "vnd.android.cursor.item/person";
   default:
    throw new IllegalArgumentException("this is unkown uri:" + uri);
  }
 }

@Override
 public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
 {
  SQLiteDatabase db = dbHelper.getReadableDatabase();
  switch (MATCHER.match(uri))
  {
   case CODE_NOPARAM:
    return db.query("person", projection, selection, selectionArgs, null, null, sortOrder);
   case CODE_PARAM:
    long id = ContentUris.parseId(uri); // 取得跟在URI后面的数字
    String where = "id = " + id;
    if (null != selection && !"".equals(selection.trim()))
    {
     where += " and " + selection;
    }
    return db.query("person", projection, where, selectionArgs, null, null, sortOrder);
   default:
    throw new IllegalArgumentException("this is unkown uri:" + uri);
  }
 }

}

测试代码
package cn.xy.test.test;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.test.AndroidTestCase;
import android.util.Log;

public class TestProviders extends AndroidTestCase
{
 // 在执行该测试方法时需要先将还有内容提供者的项目部署到Android中,否则无法找到内容提供者
 public void testInsert()
 {
  Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person");
  ContentResolver resolver = this.getContext().getContentResolver();
  ContentValues values = new ContentValues();
  values.put("name", "xy");
  values.put("phone", "111");
  resolver.insert(uri, values); // 内部调用内容提供者的insert方法
 }

// 不带id参数的删除
 public void testDelete1()
 {
  Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person");
  ContentResolver resolver = this.getContext().getContentResolver();
  int rowAffect = resolver.delete(uri, null, null);
  Log.i("rowAffect", String.valueOf(rowAffect));
 }

// 带参数的删除,通过URI传递了id至contentProvider并可追加其他条件
 public void testDelete2()
 {
  Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person/18");
  ContentResolver resolver = this.getContext().getContentResolver();
  int rowAffect = resolver.delete(uri, "name = ?", new String[] { "XY2" }); // 在provider中手动进行了拼装
  Log.i("rowAffect", String.valueOf(rowAffect));
 }
 
 public void testUpdate()
 {
  Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person/19");
  ContentResolver resolver = this.getContext().getContentResolver();
  ContentValues values = new ContentValues();
  values.put("name", "newxy");
  values.put("phone", "new111");
  int rowAffect = resolver.update(uri, values, null, null);
  Log.i("rowAffect", String.valueOf(rowAffect));
 }
 
 public void testQuery()
 {
  Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person/19");
  ContentResolver resolver = this.getContext().getContentResolver();
  Cursor cursor = resolver.query(uri, new String[]{"id","name","phone"}, null, null, "id asc");
  if(cursor.moveToFirst())
  {
   Log.i("query", cursor.getString(cursor.getColumnIndex("name")));
  }
  cursor.close();
 }
}

参考博客:http://www.cnblogs.com/chenglong/articles/1892029.html

http://blog.sina.com.cn/s/blog_67aaf4440101628t.html

Android开发13——内容提供者ContentProvider的基本使用的更多相关文章

  1. Android学习---通过内容提供者(ContentProvider)操作另外一个应用私有数据库的内容

    一.什么是ContentProvider? ContentProvider直译过来就是内容提供者,主要作用就是A应用提供接口给B应用调用数据,和之前介绍的sharedPreference和直接开放文件 ...

  2. Android开发14——监听内容提供者ContentProvider的数据变化

    一.提出需求 有A,B,C三个应用,B中的数据需要被共享,所以B中定义了内容提供者ContentProvider:A应用修改了B应用的数据,插入了一条数据.有这样一个需求,此时C应用需要得到数据被修改 ...

  3. [Android Pro] 监听内容提供者ContentProvider的数据变化

    转载自:http://blog.csdn.net/woshixuye/article/details/8281385 一.提出需求 有A,B,C三个应用,B中的数据需要被共享,所以B中定义了内容提供者 ...

  4. Android开发数据存储之ContentProvider详解

    转载:十二.ContentProvider和Uri详解 一.使用ContentProvider(内容提供者)共享数据 ContentProvider在android中的作用是对外共享数据,也就是说你可 ...

  5. 安卓第十四天笔记-内容提供者(ContentProvider)

    安卓第十四天笔记-内容提供者(ContentProvider) ContentProvider--内容提供者 1.ContentProvider简介 ContentProvider是不同应用程序之间进 ...

  6. Android中的内容提供者

    Android中的内容提供者 为什么需要内容提供者 为了跨程序访问数据.试想如果在App-1中创建了一个私有数据库,App-2是不能直接访问的.因为权限不够,虽然可以使用chmod 777来修改权限, ...

  7. [Android Pro] 内容提供者ContentProvider的基本使用

    一.ContentProvider简介 当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据.ContentProvider为存储和获取数据提 ...

  8. Android中内容提供者ContentProvider的详解

    1.什么是ContentProvider 首先,ContentProvider(内容提供者)是android中的四大组件之一,但是在一般的开发中,可能使用的比较少. ContentProvider为不 ...

  9. Android之内容提供者ContentProvider的总结

    本文包含以下知识点: ContentProvider Uri 的介绍 ContentResolver: 监听ContentProvider的数据改变 一:ContentProvider部分 Conte ...

随机推荐

  1. HDU 1495 很可乐 (DFS)

    题目链接:很可乐 解析:一个瓶子,容量为s.两个杯子,容量分别为n和m,问最少多少次倾倒才干将一瓶可乐均分为两份. 直接模拟每次的倾倒.然后递归求解. 能够加个预判的条件,要是s是奇数的时候,不管怎样 ...

  2. Java程序调用带参数的shell脚本返回值

    Java程序调用带参数的shell脚本返回值 首先来看看linux中shell变量(\(#,\)@,$0,$1,\(2)的含义解释 变量说明: -  \)$  Shell本身的PID(ProcessI ...

  3. Jmeter-Maven-Plugin高级应用:Configuring the jvm that the jmeter process runs in

    Configuring the jvm that the jmeter process runs in The JMeter Maven plugin will run the JMeter proc ...

  4. 用Bluepages来验证intranetId和Password的有效性

    代码很简单,如下: int ret = -1;ReturnCode rc = null;            cwa2 cw = new cwa2();rc = cw.authenticate(in ...

  5. 微信小程序 - 获取所在位置(省、市、区)

    实现步骤 1. 获取当前经纬度 2. 调用腾讯(百度.高德)地图对应的请求地址,一般都会有独一的key, 譬如 腾讯地图调用地址: https://apis.map.qq.com/ws/geocode ...

  6. HTML一(简介)

    什么是 HTML? HTML 是用来描述网页的一种语言. HTML 指的是超文本标记语言 (Hyper Text Markup Language) HTML 不是一种编程语言,而是一种标记语言 (ma ...

  7. Windows平台Hadoop编译、安装、配置与运行(转)

    http://www.srccodes.com/p/article/38/build-install-configure-run-apache-hadoop-2.2.0-microsoft-windo ...

  8. vue 项目的开发流程

    1.$ node -v (检测node版本,node版本需要在 V4 以上) 2.全局安装vue $ npm install -g vue 3.安装脚手架 $ npm install -g vue-c ...

  9. HDU 4847 陕西邀请赛A(水)

    HDU 4847 Wow! Such Doge! pid=4847" style="">题目链接 题意:给定文本,求有几个doge,不区分大写和小写 思路:水题.直 ...

  10. openerp发送给群组信息

    发送给群组 self.pool.get('mail.group').message_post(cr, uid, [1],             body=_('Welcome to ! Please ...