android中的ContentProvider实现数据共享

为了在应用程序之间交换数据,android中提供了ContentProvider,ContentProvider是不同应用程序之间进行数据交换的标准API。当一个应用程序需要把自己的数据暴露给其他程序使用时,该应用程序可以通过提供contentProvider来实现,其他应用程序可通过ContentResolver来操作ContentProvider暴露的数据。

ContentProvider是android应用中的四大组件之一,使用时需要在AndroidManifest.xml文件中进行配置。

一旦某个应用程序通过ContentProvider暴露了自己的数据操作接口,那么不管该应用程序是否启动,其他的应用程序都可以通过该操作接口来操作程序内部的数据,包括增删改查。

ContentProvider是不同应用程序之间进行数据交换的标准API,ContentProvider以某种Uri的形式对外提供数据,允许其他的应用访问或者修改数据。其他的应用程序使用ContentResolver根据Uri去访问操作指定的数据。

开发一个ContentProvider的步骤:

1、定义自己的ContentProvider类,该类要继承android提供的ContentProvider基类

2、在AndroidManifest.xml清单文件中对ContentProvider进行注册

 <provider android:name=".TestProvider" android:authorities="com.test.testProvider"/>


自定义的ContentProvider类继承android中提供的ContentProvider基类后还需要实现下面几个方法才能实现对外暴露数据:

onCreate() :该方法在ContentProvider创建后被调用,,当其他的应用程序第一次访问ContentProvider时,该contentProvider会被创建出来,并立即调用该onCreate()方法。适合做初始化。

query(Uri uri, String[] projection, String selection,    String[] selectionArgs, String sortOrder) :根据Uri查询满足selection条件的数据,其中projection是列名列表,表明只选择指定的数据列。

getType(Uri uri):返回Uri多代表的数据的MIME类型。如果Uri对应的数据包括多条记录,那么MIME类型的字符串应该返回以vnd.android.cursor.dir/开头,如果是一条数据那么返回的MIME字符串应该以vnd.android.cursor.item/开头

insert(Uri uri, ContentValues values) :根据Uri插入values对应的额数据

delete(Uri uri, String selection, String[] selectionArgs) :根据Uri删除和selection条件匹配的数据

update(Uri uri, ContentValues values, String selection,    String[] selectionArgs) :根据Uri更新匹配条件的数据

Uri介绍

 android中的Uri:

content://com.test.testProvider/words

Uri分为三部分:

content:// : 这个部分是android所规定的的,是固定写法。

com.test.testProvider :这个部分是ContentProvider的authorities(),系统就是由这个部分找到需要操作那个ContentProvider。只要访问指定的ContentProvider,这个部分是固定的。

words:资源部分,当访问不同的资源时,这个部分是动态改变的。

android中Uri的丰富功能:

content://com.test.testProvider/word/2

此时他要访问的资源为word/2,代表访问word数据中ID为2的记录

content://com.test.testProvider/word/2/word

此时他要访问的资源为word/2,这意味着访问word数据中ID为2的记录的word字段

content://com.test.testProvider/words   //访问全部的数据

数据为XML格式的Uri

content://com.test.testProvider/word/detail/

代表访问word节点下的detail节点

android中将字符串转化为Uri的工具类Uri:

Uri uri = Uri.parse("content://com.test.testProvider/word/2/word"); 

ContentResolver操作数据

Context提供了如下的方法来获取ContentResolver对象:getContentResolver()

ContentResolver提供如下的方法操作数据:

insert(Uri url, ContentValues values):向Uri对应的ContentProvider中插入values数据

delete(Uri url, String where, String[] selectionArgs):删除Uri对应的ContentProvider中匹配where条件的数据

update(Uri uri, ContentValues values, String where, String[] selectionArgs):更新Uri对应的ContentProvider中where条件匹配的数据

query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder):查询Uri对应的ContentProvider中匹配where条件的数据

一般来说ContentProvider是单例模式的,当多个应用程序通过ConentResolver来操作ContentProvider提供的数据时,ConentResolver调用的数据操作都将会委托给同一个ContentProvider处理。

为了确定ContentProvider实际能匹配的Uri,以及确定每个方法中Uri参数所操作的数据,android中提供了UriMatcher工具类:

addURI(String authority, String path, int code):该方法用于向UriMatcher对象注册Uri,其中authority和path组合成一个Uri,code则代表该Uri对应的标识码。

match(Uri uri):根据前面注册的Uri来判断指定的Uri对应的标示码,如果找不到匹配标识码则返回-1。

UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
matcher.addURI("com.test.testProvider", "words", 1);
matcher.addURI("com.test.testProvider", "word/#", 2);

上面的#位通配符

// 返回1
matcher.match(Uri.parse("content://com.test.testProvider/words"));
// 返回2
matcher.match(Uri.parse("content://com.test.testProvider/word/2"));
// 返回2
matcher.match(Uri.parse("content://com.test.testProvider/word/22"));


android中还提供了一个ContentUris工具类,用于操作Uri字符串:

Uri withAppendedId(Uri contentUri, long id):添加id部分

parseId(Uri contentUri):解析出包含id

监听ContentProvider的数据变化:

ContentObserver:当ContentProvider中的数据改变时调用如下的代码

getContext().getContentResolver().notifyChange(uri, null);

这行代码用于通知所有注册在该Uri上的监听者:该ContentProvider所共享的数据发生了改变

为了监听ContentProvider数据的改变,需要利用android中提供的ContentObserver基类。监听ContentProvider数据改变的监听器需要继承ContentObserver类,并重写该基类所定义的onChange(boolean selfChange)方法——当他所监听的ContentProvider的数据发生改变时,该onChange将会被调用。为了监听ContentProvider的数据变化,需要通过ContentResolver向指定的Uri注册ContentObserver监听器。ContentResolver提供如下的方法来注册监听器:

registerContentObserver(Uri uri, boolean notifyForDescendents,ContentObserver observer)

uri:该监听器所监听的ContentProvider的Uri
notifyForDescendents:如果该参数为true,假如监听的Uri为content://abc,那么Uri为content://abc/xyz、content://abc/xyz/foo的数据改变也会触发该监听器,如果为false,那么只有Uri为content://abc的数据发生了变化才会触发该监听器。

getContentResolver().registerContentObserver(uri, notifyForDescendents, observer)
原文地址:https://www.cnblogs.com/heml/p/3569452.html