1.ContentProvider是什么?
ContentProvider——内容提供者。它是一个类,这个类主要是对Android系统中进行共享的数据进行包装,并提供了一组统一的访问接口供其他程序调用。这些被共享的数据,可以使系统自己的也可以使我们个人应用程序中的数据,ContentProvider使用表的形式来组织数据.
2.为什么要有ContentProvider?
在Android中,数据的存储有很多种方式,最常用的就是SQLite和XML文件方式。在不同的应用程序间,其实数据是不能直接被相互访问和操作的,在这种情况下,ContentProvider很好的被用来解决了不同应用程序间数据共享的问题。
其实在Android系统中,已经为我们提供了许多ContentProvider,如:Contacts、Browser、CallLog、 Settings等等。那么,Android系统中提供了这么多的ContentProvider,另外还有我们自己公开的共享数据,我们在写程序的时 候,怎么才能让我们的应用程序知道去哪儿取、如何取这些数据呢?我们自然的会想到URI。
一个ContentProvider类实现了一组标准的方法接口,从而能够让其他的应用保存或读取此ContentProvider的各种数据类型。
也就是说,一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据暴露出去。
外界根本看不到,也不用看到这个应用暴露的数据在应用当中是如何存储的,或者是用数据库存储还是用文件存储,还是通过网上获得,这些一切都不重要,
重要的是外界可以通过这一套标准及统一的接口和程序里的数据打交道,可以读取程序的数据,也可以删除程序的数据
3.URI是什么?
URI(Uniform Resource Identifier)——统一资源定位符,URI在ContentProvider中代表了要操做的数据。
在Android系统中通常的URI格式为:content://com.wirelessqa.content.provider/profile/10
在万维网访问时通常用的URI格式为:/AAA/123
content://——schema,这个是Android中已经定义好的一个标准。我个人一直认为这和我们的http://有异曲同工之妙,都是代表的协议。ContentProvider(内容提供者)的scheme已经由Android所规定为:content://com.wirelessqa.content.provider——authority(主机名),用于唯一标识这个ContentProvider,外部调用者通过这个authority来找到它。相当于, 代表的是我们ContentProvider所在的”域名”,这个”域名”在我们Android中一定要是唯一的,否则系统怎么能知道该找哪一个 Provider呢?所以一般情况下,建议采用完整的包名加类名来标识这个ContentProvider的authority。/profile/10——路径,用来标识我们要操作的数据。/profile/10表示的意思是——找到profile中id为10的记录。其实这个相当于/AAA/123。
【扩展阅读】
1.要操作profile表中id为10的记录,可以构建这样的路径:/profile/10
2.要操作profile表中id为10的记录的name字段,profile/10/name
3.要操作profile表中的所有记录,可以构建这样的路径:/profile
4.要操作xxx表中的记录,可以构建这样的路径:/xxx
5.当然要操作的数据不一定来自数据库,也可以是文件、xml或网络等其他存储方式,如下:要操作xml文件中profile节点下的name节点,可以构建这样的路径:/profile/name
6.如果要把一个字符串转换成Uri,可以使用Uri类中的parse()方法,如下:Uri uri = Uri.parse(“content://com.wirelessqa.content.provider/pofile”)
综上所述,content://com.wirelessqa.content.provider/profile/10/User所代表的URI的意思为:标识com.wirelessqa.content.provider中proifle表中_ID为10的User项。
4.URI常用方法有哪些?
UriMatcher:用于匹配Uri,它的用法如下:
1. 首先把你需要匹配Uri路径全部给注册上,如下:
2.注册完需要匹配的Uri后,就可以使用uriMatcher.match(uri)方法对输入的Uri进行匹配,如果匹配就返回匹配码,匹配码是调用 addURI()方法传入的第三个参数,假设匹配 content://com.wirelessqa.content.provider/profile路径,返回的匹配 码为1。
ContentUris:用于获取Uri路径后面的ID部分,它有两个比较实用的方法:
ContentResolver:当外部应用需要对ContentProvider中的数据进行添加、删除、修改和查询操作时,可以使用ContentResolver 类来完成,要获取ContentResolver 对象,可以使用Activity提供的getContentResolver()方法。 ContentResolver使用insert、delete、update、query方法,来操作数据。
5.ContentProvider中公开的几个方法
public boolean onCreate():该方法在ContentProvider创建后就会被调用,Android系统运行后,ContentProvider只有在被第一次使用它时才会被创建。public Uri insert(Uri uri, ContentValues values):外部应用程序通过这个方法向 ContentProvider添加数据。 uri—— 标识操作数据的URIvalues—— 需要添加数据的键值对public int delete(Uri uri, String selection, String[] selectionArgs):外部应用程序通过这个方法从 ContentProvider中删除数据。 uri——标识操作数据的URIselection——构成筛选添加的语句,如”id=1″ 或者 “id=?”selectionArgs——对应selection的两种情况可以传入null 或者 new String[]{“1″}public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs):外部应用程序通过这个方法对 ContentProvider中的数据进行更新。 values——对应需要更新的键值对,键为对应共享数据中的字段,值为对应的修改值其余参数同delete方法public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder):外部应用程序通过这个方法从ContentProvider中获取数据,并返回一个Cursor对象。 projection——需要从Contentprovider中选择的字段,如果为空,则返回的Cursor将包含所有的字段。sortOrder——默认的排序规则其余参数同delete方法public String getType(Uri uri):该方法用于返回当前Url所代表数据的MIME类型。
如果操作的数据属于集合类型,那么MIME类型字符串应该以vnd.android.cursor.dir/开头,
例如:要得到所有person记录的Uri为content://com.wirelessqa.content.provider/profile,那么返回的MIME类型字符串应该为:”vnd.android.cursor.dir/profile”。
如果要操作的数据属于非集合类型数据,那么MIME类型字符串应该以vnd.android.cursor.item/开头,
例如:得到id为10的person记录,Uri为content://com.wirelessqa.content.provider/profile/10,那么返回的MIME类型字符串为:”vnd.android.cursor.item/profile”。
6.实现步骤
想要自己的程序里面的数据能够被其他程序所访问到,有以下步骤:
第一:首先生成一个继承contentprovider的类.
第二:在androidMainfest.xml里面添加一个provider的标签就可以了.
是不是很简单?其他程序访问的时候只要按以下步骤就可以访问到了:
AUTHORY其实就是 android:authorities的值.,注意.这里必须一样..否则系统是找不到的.也是就是String AUTHORY=”content://com.wirelessqa.content.provider”
然后获取一个ContentResolver mContentResolver=getContentResolver();这样就其他程序就可以反问我们的数据了
ContentResolver对应的几个方法:
query(Uri, String[], String, String[], String)which returns data to the callerinsert(Uri, ContentValues)which inserts new data into the content providerupdate(Uri, ContentValues, String, String[])which updates existing data in the content providerdelete(Uri, String, String[])which deletes data from the content provider
其实和contentprovider里面的方法是一样的..他们所对应的数据,最终是会被传到我们在之前程序里面定义的那个contentprovider类的方法,至于你想要在这几个方法里面做什么事,随你
7.实例讲解
AndroidManifest.xml
在AndroidManifest.xml的<application>和</application>之间加入:
Profile.java
DBHelper.java
MyProvider.java
定义ContentProvider的子类
1. contentprovider的调用者有可能是Activity,Service,Application这3种context,被谁调用getContext就是谁
2. ContentResolver是属于context的,通过getContentResolver获取,ContentResolver可以通过registerContentObserver注册观察者(观察者是ContentObserver 的派生类),一旦ContentProvider操作的数据变化后,调用ContentResolver的notifyChange方法即可通知到观察者(回调观察者的onChange方法),注册观察者不是必须的,所有notifyChange不是必须调用的
3. ContentValues 一次只能放入一行数据(可以使多个字段,即多个名值对)
4. onCreate只在ContentProvider第一次被调用的时候调用,多次调用共享的是一个ContentProvider
5. Cursor即数据库查询结果的操作游标,用户随机访问查询结果,获取查询结果的数目等
6. 在Content Resolver中有几个需要注意的接口:
notifyChange (Uri uri, ContentObserver observer, boolean syncToNetwork);
registerContentObserver (Uri uri, boolean notifyForDescendents, ContentObserver observer);
unregisterContentObserver (ContentObserver observer);
7.在Cursor中也有几个类似的接口:
setNotificationUri (ContentResolver cr, Uri uri);
registerContentObserver (ContentObserver observer);
unregisterContentObserver (ContentObserver observer);
8.要在query中调用
setNotificationUri(ContentResolver cr, Uri notifyUri)从而在ContentService中注册contentservice的观察者,这个观察者是cursor的内部成员(cursor是一个接口,此处真正的cursor是sqlitecursor),这样每个查询返回的cursor都能在contentprovider对应数据改变时得到通知,因为这些cursor都有一个成员注册成了contentservice的观察者
那么数据改变时,是怎么通知这些cursor的观察者成员呢?
这就需要在delete,update,insert这些改变数据的方法中调用contentresolver的notifychange方法,这个notifychange实际调用的是contentservice的notifychange,在这个notifychange方法里,contentservice查找所有在其中注册的观察者,找出对这次更新数据感兴趣的观察者(通过uri),然后通知它们数据改变
contentservice有很多观察者,它是系统服务,管理系统中所有contentprovider,通过uri匹配查找观察者,通知符合要求的观察者(实际就是通知对应的cursor)
cursor得到通知以后做些什么呢?
事实上,cursor也可以有很多观察者,因为一个查询出来的结果集可能会被多个地方使用(比如多个listview使用一个cursor),cursor对应的数据改变的时候,它也会通知到所有关注它的观察者(调用它们的onchange)
那么,cursor的观察者是怎么注册进去的呢?
是通过cursor的registerContentObserver这个方法注册进去的
以下例作为解析,在simplecursoradapter(继承自cursoradapter)的构造函数中,调用父类cursoradapter的构造函数,在这个构造函数里,
调用了cursor的registerContentObserver,把一个继承自contentobserver的成员对象作为观察者注册进了cursor的观察者里。这样cursor变化了,就会通知simplecursoradapter,simplecursoradapter里就可以重新查询结果并显示在listview中
以上其实就是两个观察者模式,cursor观察contentservice,同时cursor又被cursoradapter观察(都是通过其成员变量观察,不是直接观察),我们也可以通过contentservice和cursoradapter提供的接口注册我们自己的观察者,也就是说contentservice的观察者可以不是cursor,cursor的观察者可以不是cursoradapter
MainActivity.java
此文参考多份网络上的文章
源码下载:/detail/wirelessqa/5091983
本文链接:【Android数据存储】ContentProvider详细介绍(附实例源码)
转载声明:本站文章若无特别说明,皆为原创,转载请注明来源:WirelessQA,谢谢!^^