Android Content Provider详解

移动开发 Android
Content Provider(内容提供器)用来管理和共享应用程序的数据库。在应用程序间,Content Provider是共享数据的首选方式。这意味着,你可以配置自己的Content Provider去存取其他的应用程序或者通过其他应用程序暴露的Content Provider去存取它们的数据。Android设备本身包含了几个Content Provider来访问像联系人信息等有用的数据库。

Android中的Contentprovider机制可支持在多个应用中存储和读取数据。这也是跨应用共享数据的唯一方式。在android系统中,没有一个公共的内存区域,供多个应用共享存储数据。

Android提供了一些主要数据类型的ContentProvider,比如音频、视频、图片和私人通讯录等。可在android.provider包下面找到一些android提供的Contentprovider。可以获得这些Contentprovider,查询它们包含的数据,当然前提是已获得适当的读取权限。

如果想公开自己的数据,那么可有两种办法:

创建自己的Contentprovider,需要继承ContentProvider类; 如果你的数据和已存在的Contentprovider数据结构一致,可以将数据写到已存在的Contentprovider中,当然前提是获取写该Contentprovider的权限。比如把OA中的成员通讯信息加入到系统的联系人Contentprovider中。

所有Contentprovider都需要实现相同的接口用于查询Contentprovider并返回数据,也包括增加、修改和删除数据。

首先需要获得一个ContentResolver的实例,可通过Activity的成员方法getContentResovler()方法:

  1. ContentResolver cr = getContentResolver(); 

ContentResolver实例带的方法可实现找到指定的Contentprovider并获取到Contentprovider的数据。

ContentResolver的查询过程开始,Android系统将确定查询所需的具体Contentprovider,确认它是否启动并运行它。android系统负责初始化所有的Contentprovider,不需要用户自己去创建。实际上,contentprovider的用户都不可能直接访问到contentprovider实例,只能通过ContentResolver在中间代理。

数据模型

Contentprovider展示数据类似一个单个数据库表。其中:

每行有个带唯一值的数字字段,名为_ID,可用于对表中指定记录的定位;Contentprovider返回的数据结构,是类似JDBC的ResultSet,在android中,是Cursor对象。 URI

每个contentprovider定义一个唯一的公开的URI,用于指定到它的数据集。一个contentprovider可以包含多个数据集(可以看作多张表),这样,就需要有多个URI与每个数据集对应。这些URI要以这样的格式开头:

content://

表示这个uri指定一个contentprovider。

如果你想创建自己的contentprovider,***把自定义的URI设置为类的常量,这样简化别人的调用,并且以后如果更新URI也很容易。android定义了CONTENT_URI常量用于URI,比如:

android.provider.Contacts.Phones.CONTENT_URI
android.provider.Contacts.Photos.CONTENT_URI

要注意的是上面例子中的Contacts,已经在android 2.0及以上版本不赞成使用。

查询Contentprovider

要想使用一个contentprovider,需要以下信息:

定义这个contentprovider的URI 返回结果的字段名称 这些字段的数据类型

如果需要查询contentprovider数据集的特定记录(行),还需要知道该记录的ID的值。

构建查询

查询就是输入URI等参数,其中URI是必须的,其他是可选的,如果系统能找到URI对应的contentprovider将返回一个Cursor对象。

可以通过ContentResolver.query()或者Activity.managedQuery()方法。两者的方法参数完全一样,查询过程和返 回值也是相同的。区别是,通过Activity.managedQuery()方法,不但获取到Cursor对象,而且能够管理Cursor对象的生命周 期,比如当Activity暂停(pause)的时候,卸载该Cursor对象,当Activity restart的时候重新查询。另外,也可以对一个没有处于Activity管理的Cursor对象做成被Activity管理的,通过调用 Activity.startManaginCursor()方法。

类似这样:

  1. Cursor cur = managedQuery(myPerson, nullnullnullnull); 

其中***个参数myPerson是Uri类型实例。

如果需要查询的是指定行的记录,需要用_ID值,比如ID值为23,URI将是类似:

content://. . . ./23

android提供了方便的方法,让开发者不需要自己拼接上面这样的URI,比如类似:

  1. Uri myPerson = ContentUris.withAppendedId(People.CONTENT_URI, 23); 

或者:

  1. Uri myPerson = Uri.withAppendedPath(People.CONTENT_URI, "23"); 

二者的区别是一个接收整数类型的ID值,一个接收字符串类型。

其他几个参数:

names,可以为null,表示取数据集的全部列,或者声明一个String数组,数组中存放列名称,比如:People._ID。一般列名都在该Contentprovider中有常量对应; 针对返回结果的过滤器,格式类似于SQL中的WHERE子句,区别是不带WHERE关键字,如果返回null表示不过滤,比如name=?; 前面过滤器的参数,是String数组,是针对前面条件中?占位符的值; 排序参数,类似SQL的ORDER BY字句,不过不需要写ORDER BY部分,比如name desc,如果不排序,可输入null。

返回值是Cursor对象,游标位置在***条记录之前。

下面实例适用于android 2.0及以上版本,从android通讯录中得到姓名字段:

  1. Cursor cursor = getContentResolver().query( 
  2.         ContactsContract.CommonDataKinds.Phone.CONTENT_URI, nullnull,null,null); 

返回值的内容

返回值的内容类似上图,不同的contentprovider会有不同的列和名称,但是会有两个相同的列,上面提到过的一个是_ID,用于唯一标识记录,还有一个_COUNT,用于记录整个结果集的大小,可以看到上面图中的_COUNT的值是相同的。

读取返回的数据

如 果在查询的时候使用到ID,那么返回的数据只有一条记录。在其他情况下,一般会有多条记录。和JDBC的ResultSet类似,需要操作游标遍历结果 集,在每行,再通过列名获取到列的值,可以通过getString()、getInt()、getFloat()等方法获取值。比如类似下面:

  1. while (cursor.moveToNext()) { 
  2.     builder 
  3.             .append( 
  4.                     cursor 
  5.                             .getString(cursor 
  6.                                     .getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME))) 
  7.             .append("-"); 

和JDBC中不同,没有直接通过列名获取列值的方法,只能先列名获取到列的整型索引值,然后再通过该索引值定位获取列的值。

编辑数据

可以通过contentprovider实现以下编辑功能:

增加新的记录; 在已经存在的记录中增加新的值; 批量更新已经存在的多个记录; 删除记录。

所有的编辑功能都是通过ContentResolver的方法实现。一些Contentprovider对权限要求更严格一些,需要写的权限,如果没有会报错。

增加记录

要想增加记录到contentprovider,首先,要在ContentValues对象中设置类似map的键值对,在这里,键的值对应contentprovider中的列的名字,键值对的值,是对应列希望的类型。然后,调用ContentResolver.insert()方法,传入这个ContentValues对象,和对应Contentprovider的URI即可。返回值是这个新记录的URI对象。这样你可以通过这个URI获得包含这条记录的Cursor对象。比如:

  1. ContentValues values = new ContentValues(); 
  2.  
  3. values.put(People.NAME, "Abraham Lincoln"); 
  4.  
  5. Uri uri = getContentResolver().insert(People.CONTENT_URI, values); 

在原有记录上增加值

如果记录已经存在,可在记录上增加新的值,或者编辑已经存在的值。

首先要过去到原来的值对象,然后要清除原有的值,然后像上面增加记录一样即可:

  1. Uri uri=Uri.withAppendedPath(People.CONTENT_URI, "23"); 
  2.  
  3. Uri phoneUri = Uri.withAppendedPath(uri, People.Phones.CONTENT_DIRECTORY); 
  4.  
  5. values.clear(); 
  6. values.put(People.Phones.TYPE, People.Phones.TYPE_MOBILE); 
  7. values.put(People.Phones.NUMBER, "1233214567"); 
  8. getContentResolver().insert(phoneUri, values); 

批量更新值

批量更新一组记录的值,比如NY改名为Eew York。可调用ContenResolver.update()方法。

删除记录

如果是删除单个记录,调用ContentResolver.delete()方法,URI参数,指定到具体行即可。

如果是删除多个记录,调用ContentResolver.delete()方法,URI参数指定Contentprovider即可,并带一个类似SQL的WHERE子句条件。这里和上面类似,不带WHERE关键字。

创建自己的Contentprovider

创建contentprovider,需要:

设置存储系统。大多数contentprovider使 用文件或者SQLite数据库,不过你可以用任何方式存储数据。android提供SQLiteoOpenHelper帮助开发者创建和管理 SQLiteDatabase。 继承ContentProvider,提供对数据的访问。 在manifest文件中声明contentprovider。 继承ContentProvider类

必须定义ContentProvider类的子类,需要实现如下方法:

query()
insert()
update()
delete()
getType()
onCreate()

query() 方法,返回值是Cursor实例,用于迭代请求的数据。Cursor是一个接口。android为该接口提供了一些只读的(和JDBC的 ResultSet不一样,后者还提供可写入的可选特性)Cursor实现。比如SQLiteCursor,可迭代SQLite数据库中的数据。可以通过 SQLiteDatabase类的query()方法获取到该Cursor实例。还有其他的Cursor实现,比如MatrixCursor,用于数据不 是存储在数据库的情况下。

因为Contentprovider可能被多个ContentResolver对象在不同的进程和线程中调用,因此实现Contentprovider必须考虑线程安全问题。

作为良好的习惯,在实现编辑数据的代码中,要调用ContentResolver.notifyChange()方法,通知那些监听数据变化的监听器。

在实现子类的时候,还有一些步骤可以简化Contentprovider客户端的使用:

定义public static final Uri常量,名称为CONTENT_URI:

  1. public static final UriCONTENT_URI = 
  2.                Uri.parse("content://com.example.codelab.transportationprovider"); 

如果有多个表,它们也是使用相同的CONTENT_URI,只是它们的路径部分不同。

也就是说红色框部分是一致的。

定义返回的列名,public static final,列名的值,比如使用SQLite数据库作为存储,对应表的列名。

在文档中要写出各个列的数据类型,便于使用者读取。

如果需要处理新的MIME数据类型,比如通过Intent的方式,并且带data的mimeType,那么需要在ContentProvider.getType()方法中进行处理,参见编写完整的Contentprovider示例编写一个getType方法部分。

如果处理数据库表中超大的数据,比如很大的位图文件,一般存在文件系统中,可以参照在contentprovider中使用大型二进制文件,这样第三方的contentprovider使用者,可以访问不属于它权限的文件,通过contentprovider做代理。

声明ContentProvider

创建ContentProvider后,需要在manifest文件中声明,android系统才能知道它,当其他应用需要调用该ContentProvider时才能创建或者调用它。

语法类似:

  1. <provider android:name="com.easymorse.cp.MyContentProvider"  
  2.           android:authorities="com.easymorse.cp.mycp"></provider>  

android:name要写ContentProvider继承类的全名。

android:authorities要写和CONTENT_URI常量的B部分(见上面图)。

注意不要把上图C和D部分加到authorities中去。authorities是用来识别ContentProvider的,C和D部分实际上是ContentProvider内部使用的。

责任编辑:徐川 来源: net
相关推荐

2009-11-18 16:43:59

2023-04-13 07:52:59

2009-07-21 10:40:36

ASP.NET Pro

2014-07-28 10:09:30

Android

2011-05-27 15:02:15

Android ListView

2013-11-14 10:42:48

MTPAndroid

2017-01-11 19:05:45

AndroidAndroid Loa详解

2014-07-24 09:11:34

2013-11-14 16:50:08

2011-08-01 10:01:25

SQLite

2009-11-18 16:51:21

Oracle Prov

2013-12-25 09:34:26

Android SDKAndroid组件

2021-09-07 08:49:35

Android

2010-07-13 09:02:19

Widget开发

2011-09-09 20:14:58

Android Wid

2013-01-11 13:48:41

Android开发组件Notificatio

2009-12-30 08:52:17

Ubuntu Tora

2009-07-21 14:37:13

Profile Pro优化ASP.NET 2

2021-09-07 09:53:45

鸿蒙HarmonyOS应用

2013-01-05 09:21:55

Ubuntu for UbuntuAndroid
点赞
收藏

51CTO技术栈公众号