ContentProvider和Uri详解

简介:

一、使用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使用<provider>对该ContentProvider进行配置,为了能让其他应用找到该ContentProvider ,ContentProvider采用了authorities(主机名/域名)对它进行唯一标识,你可以把ContentProvider看作是一个网站(想想,网站也是提供数据者),authorities 就是他的域名:

< manifest....  >
   
< application  android:icon ="@drawable/icon"  android:label ="@string/app_name" >
      
< provider  android:name =".PersonContentProvider"  
           android:authorities
="com.ljq.providers.personprovider" />
   
</ application >
</ manifest >

 

二、Uri介绍

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

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")

                

             

三、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

                                    

四、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 

五、使用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"。

              

                

六、使用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, nullnullnull"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, nullnull);  
  18.  //删除id为2的记录Uri   
  19.  deleteIdUri = ContentUris.withAppendedId(uri, 2);  
  20.  resolver.delete(deleteIdUri, nullnull); 

七、监听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"), truenew PersonObserver(new Handler()));  
  2.     public class PersonObserver extends ContentObserver{     
  3.         public PersonObserver(Handler handler) {        
  4.             super(handler);     
  5.             }     
  6.         public void onChange(boolean selfChange){        
  7.             //此处可以进行相应的业务处理   }}  
  8.         }  
  9.     } 

 


     本文转自xyz_lmn51CTO博客,原文链接:http://blog.51cto.com/xyzlmn/817105,如需转载请自行联系原作者

相关文章
|
11月前
|
API 数据库 数据库管理
ContentProvider初探
ContentProvider初探
43 0
|
数据库
ContentProvider
构建content URI public class TaskContract { /* COMPLETED (1) Add content provider constants to the Contract Clients need to know how to access the task data, and it's your job to provide these content URI's for the path to that data: 1) Content authority, 2) Base content
111 0
ContentProvider
|
存储 安全 API
透过FileProvider再看ContentProvider
大家应该都熟悉FileProvider吧,但是其诞生的原因,内部怎么实现的,又是怎么转化为文件的,大家有了解多少呢?今天就通过它重新看看ContentProvider这个四大组件之一。
329 0
$request_uri和$uri
$request_uri This variable is equal to the *original* request URI as received from the client including the args. It cannot be modified. Look at $uri for the post-rewrite/altered URI. Does not in
1441 0
|
XML Android开发 数据格式