在Android中,ContentProvider是一种数据包装器,适合在不同进程间实现信息的共享。

例如,在Android中SQLite数据库是一个典型的数据源,我们可以把它封装到ContentProvider中,这样就可以很好的为其他应用提供信息共享服务。其他应用在访问ContentProvider时,可以使用一组类似REST的URI的方式进行数据操作,大大简化了读写信息的复杂度。例如,如果要从封装图书数据库的ContentProvider获取一组图书,需要使用类似以下形式的URI:

content://com.scott.book.BookProvider/books

而要从图书数据库中获取指定图书(比如23号图书),需要使用类似以下形式的URI:

content://com.scott.book.BookProvider/books/23

注:ContentProvider是一个抽象类,定义了一系列操作数据的方法模板,BookProvider需要实现这些方法,实现图书信息的各种操作。

那么,现在知道了具体的URI之后,我们又如何操作进而取得数据呢?

此时,我们就要了解ContentResolver这个类,它跟ContentProvider是对应的关系,我们正是通过它来与ContentProvider进行数据交换的。android.content.Context类为我们定义了getContentResolver()方法,用于获取一个ContentResolver对象,如果我们在运行期可以通过getContext()获取当前Context实例对象,就可以通过这个实例对象所提供的getContentResolver()方法获取到ContentResolver类型的实例对象,进而可以操作对应的数据。

下面我们就通过联系人实例对这种机制进行演示。

在Android中,联系人的操作都是通过一个统一的途径来读写数据的,我们打开/data/data/com.android.providers.contacts可以看到联系人的数据源:


有兴趣的朋友可以导出这个文件,用专业的工具软件打开看一下表结构。

对这个SQLite类型的数据源的封装后,联系人就以ContentProvider的形式为其他应用进程提供联系人的读写服务,我们就可以顺利成章的操作自己的联系人信息了。

为了方便测试,我们先添加两个联系人到数据源中,如图所示:



我们看到,每个联系人都有两个电话号码和两个邮箱账号,分别为家庭座机号码、移动手机号码、家庭邮箱账号和工作邮箱账号。当然在添加联系人时有很多其他信息,我们这里都没有填写,只选择了最常用的电话和邮箱,主要是方便演示这个过程。

在演示代码之前,我们需要了解一下android.provider.ContactsContract这个类(注:在较早的版本中是android.provider.Contacts这个类,不过现在已被废弃,不建议使用),它定义了各种联系人相关的URI和每一种类型信息的属性信息:


有兴趣的朋友还可以读一下源代码,不过比较多,而且内部类使用的特别多,读起来有一定的困难,还是要做好心理准备。

下面我们通过一个项目,来演示一下联系人操作的具体过程。新建一个名为provider的项目,创建一个名为ContactsReadTest的测试用例,如下:

[java] view plain copy print ?
  1. packagecom.scott.provider;
  2. importjava.util.ArrayList;
  3. importandroid.content.ContentResolver;
  4. importandroid.database.Cursor;
  5. importandroid.net.Uri;
  6. importandroid.provider.ContactsContract;
  7. importandroid.test.AndroidTestCase;
  8. importandroid.util.Log;
  9. publicclassContactsReadTestextendsAndroidTestCase{
  10. privatestaticfinalStringTAG="ContactsReadTest";
  11. //[content://com.android.contacts/contacts]
  12. privatestaticfinalUriCONTACTS_URI=ContactsContract.Contacts.CONTENT_URI;
  13. //[content://com.android.contacts/data/phones]
  14. privatestaticfinalUriPHONES_URI=ContactsContract.CommonDataKinds.Phone.CONTENT_URI;
  15. //[content://com.android.contacts/data/emails]
  16. privatestaticfinalUriEMAIL_URI=ContactsContract.CommonDataKinds.Email.CONTENT_URI;
  17. privatestaticfinalString_ID=ContactsContract.Contacts._ID;
  18. privatestaticfinalStringDISPLAY_NAME=ContactsContract.Contacts.DISPLAY_NAME;
  19. privatestaticfinalStringHAS_PHONE_NUMBER=ContactsContract.Contacts.HAS_PHONE_NUMBER;
  20. privatestaticfinalStringCONTACT_ID=ContactsContract.Data.CONTACT_ID;
  21. privatestaticfinalStringPHONE_NUMBER=ContactsContract.CommonDataKinds.Phone.NUMBER;
  22. privatestaticfinalStringPHONE_TYPE=ContactsContract.CommonDataKinds.Phone.TYPE;
  23. privatestaticfinalStringEMAIL_DATA=ContactsContract.CommonDataKinds.Email.DATA;
  24. privatestaticfinalStringEMAIL_TYPE=ContactsContract.CommonDataKinds.Email.TYPE;
  25. publicvoidtestReadContacts(){
  26. ContentResolverresolver=getContext().getContentResolver();
  27. Cursorc=resolver.query(CONTACTS_URI,null,null,null,null);
  28. while(c.moveToNext()){
  29. int_id=c.getInt(c.getColumnIndex(_ID));
  30. StringdisplayName=c.getString(c.getColumnIndex(DISPLAY_NAME));
  31. Log.i(TAG,displayName);
  32. ArrayList<String>phones=newArrayList<String>();
  33. ArrayList<String>emails=newArrayList<String>();
  34. Stringselection=CONTACT_ID+"="+_id;//the'where'clause
  35. //获取手机号
  36. inthasPhoneNumber=c.getInt(c.getColumnIndex(HAS_PHONE_NUMBER));
  37. if(hasPhoneNumber>0){
  38. Cursorphc=resolver.query(PHONES_URI,null,selection,null,null);
  39. while(phc.moveToNext()){
  40. StringphoneNumber=phc.getString(phc.getColumnIndex(PHONE_NUMBER));
  41. intphoneType=phc.getInt(phc.getColumnIndex(PHONE_TYPE));
  42. phones.add(getPhoneTypeNameById(phoneType)+":"+phoneNumber);
  43. }
  44. phc.close();
  45. }
  46. Log.i(TAG,"phones:"+phones);
  47. //获取邮箱
  48. Cursoremc=resolver.query(EMAIL_URI,null,selection,null,null);
  49. while(emc.moveToNext()){
  50. StringemailData=emc.getString(emc.getColumnIndex(EMAIL_DATA));
  51. intemailType=emc.getInt(emc.getColumnIndex(EMAIL_TYPE));
  52. emails.add(getEmailTypeNameById(emailType)+":"+emailData);
  53. }
  54. emc.close();
  55. Log.i(TAG,"emails:"+emails);
  56. }
  57. c.close();
  58. }
  59. privateStringgetPhoneTypeNameById(inttypeId){
  60. switch(typeId){
  61. caseContactsContract.CommonDataKinds.Phone.TYPE_HOME:return"home";
  62. caseContactsContract.CommonDataKinds.Phone.TYPE_MOBILE:return"mobile";
  63. caseContactsContract.CommonDataKinds.Phone.TYPE_WORK:return"work";
  64. default:return"none";
  65. }
  66. }
  67. privateStringgetEmailTypeNameById(inttypeId){
  68. switch(typeId){
  69. caseContactsContract.CommonDataKinds.Email.TYPE_HOME:return"home";
  70. caseContactsContract.CommonDataKinds.Email.TYPE_WORK:return"work";
  71. caseContactsContract.CommonDataKinds.Email.TYPE_OTHER:return"other";
  72. default:return"none";
  73. }
  74. }
  75. }
为了使这个测试用例运行起来,我们需要在AndroidManifest.xml中配置一下测试设备的声明,它与<application>元素处于同一级别位置:

[html] view plain copy print ?
  1. <!--配置测试设备的主类和目标包-->
  2. <instrumentationandroid:name="android.test.InstrumentationTestRunner"
  3. android:targetPackage="com.scott.provider"/>
然后再配置使用测试类库声明,它与<activity>元素处于同一级别位置:

[html] view plain copy print ?
  1. <!--配置测试要使用的类库-->
  2. <uses-libraryandroid:name="android.test.runner"/>
最后,还有一个重要的声明需要配置,就是读取联系人权限,声明如下:

[html] view plain copy print ?
  1. <!--读取联系人-->
  2. <uses-permissionandroid:name="android.permission.READ_CONTACTS"/>
经过以上准备工作,这个测试用例就可以运转起来了,我们运行一下testReadContacts()方法,打印结果如下:


看来联系人里的信息都被我们准确无误的读取出来了。

如果我们在一个Activity里运行读取联系人的代码,不仅可以使用ContentResolver直接进行读取操作(即查询),还可以使用Activity提供的managedQuery方法方便的实现同样的效果,我们来看一下这个方法的具体代码:

[java] view plain copy print ?
  1. publicfinalCursormanagedQuery(Uriuri,
  2. String[]projection,
  3. Stringselection,
  4. String[]selectionArgs,
  5. StringsortOrder)
  6. {
  7. Cursorc=getContentResolver().query(uri,projection,selection,selectionArgs,sortOrder);
  8. if(c!=null){
  9. startManagingCursor(c);
  10. }
  11. returnc;
  12. }
我们发现,其实它还是使用了ContentResolver进行查询操作,但是多了一步startManagingCursor的操作,它会根据Activity的生命周期对Cursor对象进行管理,避免了一些因Cursor是否释放引起的问题,所以非常方便,大大简化了我们的工作量。

接下来我们将要尝试将一个联系人信息添加到系统联系人的数据源中,实现对联系人的写入操作。我们新建一个名为ContactsWriteTest的测试用例,如下:

[java] view plain copy print ?
  1. packagecom.scott.provider;
  2. importjava.util.ArrayList;
  3. importandroid.content.ContentProviderOperation;
  4. importandroid.content.ContentProviderResult;
  5. importandroid.content.ContentResolver;
  6. importandroid.net.Uri;
  7. importandroid.provider.ContactsContract;
  8. importandroid.test.AndroidTestCase;
  9. importandroid.util.Log;
  10. publicclassContactsWriteTestextendsAndroidTestCase{
  11. privatestaticfinalStringTAG="ContactsWriteTest";
  12. //[content://com.android.contacts/raw_contacts]
  13. privatestaticfinalUriRAW_CONTACTS_URI=ContactsContract.RawContacts.CONTENT_URI;
  14. //[content://com.android.contacts/data]
  15. privatestaticfinalUriDATA_URI=ContactsContract.Data.CONTENT_URI;
  16. privatestaticfinalStringACCOUNT_TYPE=ContactsContract.RawContacts.ACCOUNT_TYPE;
  17. privatestaticfinalStringACCOUNT_NAME=ContactsContract.RawContacts.ACCOUNT_NAME;
  18. privatestaticfinalStringRAW_CONTACT_ID=ContactsContract.Data.RAW_CONTACT_ID;
  19. privatestaticfinalStringMIMETYPE=ContactsContract.Data.MIMETYPE;
  20. privatestaticfinalStringNAME_ITEM_TYPE=ContactsContract.CommonDataKinds.StructuredName.CONTENT_ITEM_TYPE;
  21. privatestaticfinalStringDISPLAY_NAME=ContactsContract.CommonDataKinds.StructuredName.DISPLAY_NAME;
  22. privatestaticfinalStringPHONE_ITEM_TYPE=ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE;
  23. privatestaticfinalStringPHONE_NUMBER=ContactsContract.CommonDataKinds.Phone.NUMBER;
  24. privatestaticfinalStringPHONE_TYPE=ContactsContract.CommonDataKinds.Phone.TYPE;
  25. privatestaticfinalintPHONE_TYPE_HOME=ContactsContract.CommonDataKinds.Phone.TYPE_HOME;
  26. privatestaticfinalintPHONE_TYPE_MOBILE=ContactsContract.CommonDataKinds.Phone.TYPE_MOBILE;
  27. privatestaticfinalStringEMAIL_ITEM_TYPE=ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE;
  28. privatestaticfinalStringEMAIL_DATA=ContactsContract.CommonDataKinds.Email.DATA;
  29. privatestaticfinalStringEMAIL_TYPE=ContactsContract.CommonDataKinds.Email.TYPE;
  30. privatestaticfinalintEMAIL_TYPE_HOME=ContactsContract.CommonDataKinds.Email.TYPE_HOME;
  31. privatestaticfinalintEMAIL_TYPE_WORK=ContactsContract.CommonDataKinds.Email.TYPE_WORK;
  32. privatestaticfinalStringAUTHORITY=ContactsContract.AUTHORITY;
  33. publicvoidtestWriteContacts()throwsException{
  34. ArrayList<ContentProviderOperation>operations=newArrayList<ContentProviderOperation>();
  35. ContentProviderOperationoperation=ContentProviderOperation.newInsert(RAW_CONTACTS_URI)
  36. .withValue(ACCOUNT_TYPE,null)
  37. .withValue(ACCOUNT_NAME,null)
  38. .build();
  39. operations.add(operation);
  40. //添加联系人名称操作
  41. operation=ContentProviderOperation.newInsert(DATA_URI)
  42. .withValueBackReference(RAW_CONTACT_ID,0)
  43. .withValue(MIMETYPE,NAME_ITEM_TYPE)
  44. .withValue(DISPLAY_NAME,"ScottLiu")
  45. .build();
  46. operations.add(operation);
  47. //添加家庭座机号码
  48. operation=ContentProviderOperation.newInsert(DATA_URI)
  49. .withValueBackReference(RAW_CONTACT_ID,0)
  50. .withValue(MIMETYPE,PHONE_ITEM_TYPE)
  51. .withValue(PHONE_TYPE,PHONE_TYPE_HOME)
  52. .withValue(PHONE_NUMBER,"01034567890")
  53. .build();
  54. operations.add(operation);
  55. //添加移动手机号码
  56. operation=ContentProviderOperation.newInsert(DATA_URI)
  57. .withValueBackReference(RAW_CONTACT_ID,0)
  58. .withValue(MIMETYPE,PHONE_ITEM_TYPE)
  59. .withValue(PHONE_TYPE,PHONE_TYPE_MOBILE)
  60. .withValue(PHONE_NUMBER,"13034567890")
  61. .build();
  62. operations.add(operation);
  63. //添加家庭邮箱
  64. operation=ContentProviderOperation.newInsert(DATA_URI)
  65. .withValueBackReference(RAW_CONTACT_ID,0)
  66. .withValue(MIMETYPE,EMAIL_ITEM_TYPE)
  67. .withValue(EMAIL_TYPE,EMAIL_TYPE_HOME)
  68. .withValue(EMAIL_DATA,"[email protected]")
  69. .build();
  70. operations.add(operation);
  71. //添加工作邮箱
  72. operation=ContentProviderOperation.newInsert(DATA_URI)
  73. .withValueBackReference(RAW_CONTACT_ID,0)
  74. .withValue(MIMETYPE,EMAIL_ITEM_TYPE)
  75. .withValue(EMAIL_TYPE,EMAIL_TYPE_WORK)
  76. .withValue(EMAIL_DATA,"[email protected]")
  77. .build();
  78. operations.add(operation);
  79. ContentResolverresolver=getContext().getContentResolver();
  80. //批量执行,返回执行结果集
  81. ContentProviderResult[]results=resolver.applyBatch(AUTHORITY,operations);
  82. for(ContentProviderResultresult:results){
  83. Log.i(TAG,result.uri.toString());
  84. }
  85. }
  86. }
在上面的代码中,我们把整个操作分为几个ContentProviderOperation操作,并将他们做批处理操作,我们也许注意到,从第二个操作开始,每一项都有一个withValueBackReference(RAW_CONTACT_ID, 0)步骤,它参照了第一项操作新添加的联系人的id,因为是批处理,我们插入数据前并不知道id的值,不过这个不用担心,在进行批处理插入数据时,它会重新引用新的id值,不会影响最终的结果。

当然,这个也不能忘了配置写入联系人的权限声明:

[html] view plain copy print ?
  1. <!--写入联系人-->
  2. <uses-permissionandroid:name="android.permission.WRITE_CONTACTS"/>
经过以上步骤之后,我们运行一下testWriteContacts()方法,看看联系人是否添加进去了:


更多相关文章

  1. Android官方命令深入分析之bmgr
  2. Android性能优化之内存篇
  3. 学习到的有关android的知识点 单元测试等(1)
  4. Android(安卓)Studio学习-连接真机测试教学
  5. Android(安卓)异步更新UI----handler+thread
  6. Android(安卓)数据库的简单使用
  7. 【Android】第4章(5) 示例--列出手机上的所有联系人
  8. Android并发修改异常:java.util.ConcurrentModificationException
  9. Android(安卓)应用安装过程分析

随机推荐

  1. 8.6数组使用中俩个常见的问题
  2. Oracle Linux 7.9+Oracle 12c+ASM安装文
  3. Oracle Linux 7.9+Oracle 12c+ASM安装文
  4. Shiro源码分析---FilterChain创建过程
  5. 用户管理
  6. Elasticsearch 之 elastalert监控告警通
  7. xhost: unable to open display "localh
  8. Kubernetes滚动更新速率控制解读
  9. Go实现简单负载均衡
  10. Thanos 简介和实践