ContentProvider和Uri详解

 

 

 

问题引出: "/getScrollData" 的使用, java

public static final Uri INBOX_CONTENT_URI_SCROLLDATA = Uriandroid

.parse("content://" + SMS_CONFIG_AUTHORITY + "/"数据库

+ SMS_INBOX_TABLE_NAME + "/getScrollData");网络

Cursor cursor = cr.query(app

SystemSMSInboxConfig.INBOX_CONTENT_URI_SCROLLDATA, null, null,ide

new String[] { String.valueOf(start), String.valueOf(end) },工具

SystemSMSInboxConfig._ID + " desc ");学习

这个查询是查询整张表, 仍是只查询剩余ID的一张表。网站

看provider实现, spa

case TYPE_SYSTEM_EVENT_LOG_LIMIT:
			return db.rawQuery("select * from "
					+ SystemSMSInboxConfig.SMS_INBOX_TABLE_NAME + " order by "
					+ sortOrder + " limit ?,? ", selectionArgs);

结论是:

使用了ID, 依然是基于整张表的,因此,能够先过滤表中数据,而后在得到指定ID值的行。

 




1、使用ContentProvider(内容提供者)共享数据

ContentProvider在android中的做用是对外共享数据, 也就是说你能够经过ContentProvider把应用中的数据共享给其余应用访问,其余应用能够经过ContentProvider对你应用中的数据 进行添删改查。关于数据共享,之前咱们学习过文件操做模式,知道经过指定文件的操做模式为Context.MODE_WORLD_READABLE或 Context.MODE_WORLD_WRITEABLE一样也能够对外共享数据。那么,这里为什么要使用ContentProvider对外共享数据 呢?是这样的,若是采用文件操做模式对外共享数据,数据的访问方式会因数据存储的方式而不一样,致使数据的访问方式没法统一,如:采用xml文件对外共享数 据,须要进行xml解析才能读取数据;采用sharedpreferences共享数据,须要使用sharedpreferences API读取数据。
使用ContentProvider对外共享数据的好处是统一了数据的访问方式。
当应用须要经过ContentProvider对外共享数据时,第一步须要继承ContentProvider并重写下面方法:


  1. public class PersonContentProvider extends ContentProvider{
  2. public boolean onCreate()
  3. public Uri insert(Uri uri, ContentValues values)
  4. public int delete(Uri uri, String selection, String[] selectionArgs)
  5. public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
  6. public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
  7. public String getType(Uri uri)
  8. }
复制代码
第二步须要在AndroidManifest.xml使用对该ContentProvider进行配置,为了能让 其余应用找到该ContentProvider ,ContentProvider采用了authorities(主机名/域名)对它进行惟一标识,你能够把ContentProvider看做是一个网 站(想一想,网站也是提供数据者),authorities 就是他的域名:
  1. <manifest.... >
  2.    <application android:icon="@drawable/icon" android:label="@string/app_name">
  3.       <provider android:name=".PersonContentProvider" 
  4.            android:authorities="com.ljq.providers.personprovider"/>
  5.    </application>
  6. </manifest>
复制代码

2、Uri介绍

Uri表明了要操做的数据,Uri主要包含了两部分信息:1》须要操做的ContentProvider ,2》对ContentProvider中的什么数据进行操做,一个Uri由如下几部分组成:         

2011052823392991.png

                 

ContentProvider(内容提供者)的scheme已经由Android所规定, scheme为:content://
主机名(或叫Authority)用于惟一标识这个ContentProvider,外部调用者能够根据这个标识来找到它。
路径(path)能够用来表示咱们要操做的数据,路径的构建应根据业务而定,以下:
要操做person表中id为10的记录,能够构建这样的路径:/person/10
要操做person表中id为10的记录的name字段, person/10/name
要操做person表中的全部记录,能够构建这样的路径:/person
要操做xxx表中的记录,能够构建这样的路径:/xxx
固然要操做的数据不必定来自数据库,也能够是文件、xml或网络等其余存储方式,以下:
要操做xml文件中person节点下的name节点,能够构建这样的路径:/person/name
若是要把一个字符串转换成Uri,可使用Uri类中的parse()方法,以下:
Uri uri = Uri.parse("content://com.ljq.provider.personprovider/person")



3、UriMatcher类使用介绍

由于Uri表明了要操做的数据,因此咱们常常须要解析Uri,并从Uri中获取数据。Android系统提供了两个用于操做Uri的工具类,分别为UriMatcher和ContentUris 。掌握它们的使用,会便于咱们的开发工做。
UriMatcher类用于匹配Uri,它的用法以下:
首先第一步把你须要匹配Uri路径所有给注册上,以下:


  1. //常量UriMatcher.NO_MATCH表示不匹配任何路径的返回码
  2. UriMatcher sMatcher = new UriMatcher(UriMatcher.NO_MATCH);
  3. //若是match()方法匹配content://com.ljq.provider.personprovider/person路径,返回匹配码为1
  4. sMatcher.addURI("com.ljq.provider.personprovider", "person", 1);//添加须要匹配uri,若是匹配就会返回匹配码
  5. //若是match()方法匹配content://com.ljq.provider.personprovider/person/230路径,返回匹配码为2
  6. sMatcher.addURI("com.ljq.provider.personprovider", "person/#", 2);//#号为通配符
  7. switch (sMatcher.match(Uri.parse("content://com.ljq.provider.personprovider/person/10"))) { 
  8. case 1
  9. break;
  10. case 2
  11. break;
  12. default://不匹配
  13. break;
  14. }
复制代码

注册完须要匹配的Uri后,就可使用sMatcher.match(uri)方法对输入的Uri进行匹配,若是匹配就返回匹配码,匹配码是调用 addURI()方法传入的第三个参数,假设匹配content://com.ljq.provider.personprovider/person路 径,返回的匹配码为1



4、ContentUris类使用介绍

ContentUris类用于操做Uri路径后面的ID部分,它有两个比较实用的方法:
withAppendedId(uri, id)用于为路径加上ID部分:


  1. Uri uri = Uri.parse("content://com.ljq.provider.personprovider/person")
  2. Uri resultUri = ContentUris.withAppendedId(uri, 10); 
  3. //生成后的Uri为:content://com.ljq.provider.personprovider/person/10
复制代码
parseId(uri)方法用于从路径中获取ID部分:

  1. Uri uri = Uri.parse("content://com.ljq.provider.personprovider/person/10")
  2. long personid = ContentUris.parseId(uri);//获取的结果为:10
复制代码

5、使用ContentProvider共享数据

ContentProvider类主要方法的做用:
public boolean onCreate():该方法在ContentProvider建立后就会被调用,Android开机后,ContentProvider在其它应用第一次访问它时才会被建立。
public Uri insert(Uri uri, ContentValues values):该方法用于供外部应用往ContentProvider添加数据。
public int delete(Uri uri, String selection, String[] selectionArgs):该方法用于供外部应用从ContentProvider删除数据。
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs):该方法用于供外部应用更新ContentProvider中的数据。
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder):该方法用于供外部应用从ContentProvider中获取数据。
public String getType(Uri uri):该方法用于返回当前Url所表明数据的MIME类型。

若是操做的数据属于集合类型,那么MIME类型字符串应该以vnd.android.cursor.dir/开头,

例如:要获得全部person记录的Uri为content://com.ljq.provider.personprovider/person,那么返回的MIME类型字符串应该为:"vnd.android.cursor.dir/person"。

若是要操做的数据属于非集合类型数据,那么MIME类型字符串应该以vnd.android.cursor.item/开头,

例如:获得id为10的person记录,Uri为content://com.ljq.provider.personprovider/person/10,那么返回的MIME类型字符串为:"vnd.android.cursor.item/person"。



6、使用ContentResolver操做ContentProvider中的数据

当外部应用须要对ContentProvider中的数据进行添加、 删除、修改和查询操做时,可使用ContentResolver 类来完成,要获取ContentResolver 对象,可使用Activity提供的getContentResolver()方法。 ContentResolver 类提供了与ContentProvider类相同签名的四个方法:
public Uri insert(Uri uri, ContentValues values):该方法用于往ContentProvider添加数据。
public int delete(Uri uri, String selection, String[] selectionArgs):该方法用于从ContentProvider删除数据。
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs):该方法用于更新ContentProvider中的数据。
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder):该方法用于从ContentProvider中获取数据。

这些方法的第一个参数为Uri,表明要操做的ContentProvider和对其中的什么数据进行操做,

假设给定的是:Uri.parse("content://com.ljq.providers.personprovider/person /10"),那么将会对主机名为com.ljq.providers.personprovider的ContentProvider进行操做,操做的数 据为person表中id为10的记录。

使用ContentResolver对ContentProvider中的数据进行添加、删除、修改和查询操做:


  1. ContentResolver resolver = getContentResolver();
  2. Uri uri = Uri.parse("content://com.ljq.provider.personprovider/person");
  3. //添加一条记录
  4. ContentValues values = new ContentValues();
  5. values.put("name", "linjiqin");
  6. values.put("age", 25);
  7. resolver.insert(uri, values); 
  8. //获取person表中全部记录
  9. Cursor cursor = resolver.query(uri, null, null, null, "personid desc");
  10. while(cursor.moveToNext()){
  11. Log.i("ContentTest", "personid="+ cursor.getInt(0)+ ",name="+ cursor.getString(1));
  12. }
  13. //把id为1的记录的name字段值更改新为zhangsan
  14. ContentValues updateValues = new ContentValues();
  15. updateValues.put("name", "zhangsan");
  16. Uri updateIdUri = ContentUris.withAppendedId(uri, 2);
  17. resolver.update(updateIdUri, updateValues, null, null);
  18. //删除id为2的记录
  19. Uri deleteIdUri = ContentUris.withAppendedId(uri, 2);
  20. resolver.delete(deleteIdUri, null, null);
复制代码

7、监听ContentProvider中数据的变化

若是ContentProvider的访问者须要知道ContentProvider中的数据发生变化,能够在ContentProvider发生 数据变化时调用getContentResolver().notifyChange(uri, null)来通知注册在此URI上的访问者,例子以下:


  1. public class PersonContentProvider extends ContentProvider {
  2. public Uri insert(Uri uri, ContentValues values) {
  3. db.insert("person", "personid", values);
  4. getContext().getContentResolver().notifyChange(uri, null);
  5. }
  6. }
复制代码
若是ContentProvider的访问者须要获得数据变化通知,必须使用ContentObserver对数据(数据采用uri描述)进行监听,当监听到数据变化通知时,系统就会调用ContentObserver的onChange()方法:

  1. getContentResolver().registerContentObserver(Uri.parse("content://com.ljq.providers.personprovider/person"),
  2. true, new PersonObserver(new Handler()));
  3. public class PersonObserver extends ContentObserver{
  4. public PersonObserver(Handler handler) {
  5. super(handler);
  6. }
  7. public void onChange(boolean selfChange) {
  8. //此处能够进行相应的业务处理
  9. }
  10. }
相关文章
相关标签/搜索