android ContentProvider

本是古典 何须时尚 2022-08-09 17:56 278阅读 0赞

个人原创,请勿copy,转发请注明出处http://blog.csdn.net/zenmela2011/article/details/49616339

ContentProvider(内容提供器)主要用于在不同的应用程序之间实现数据共享的功能。他提供了一套完整的机制,允许一个程序访问另一个程序中的数据,同时还能保证被访数据的安全性。目前,使用内容提供器是android实现跨程序共享数据的标准方法。

内容提供器可以选择只对哪一部分数据进行共享,从而保证我们程序中的隐私数据不会有泄露的风险。

要访问内容提供器中共享的数据,就一定要借助ContentResolver类,可以通过Context中的getContentResolver()方法获得该类的实例。ContentResolver中提供了insert()、delete()、update()、query()方法来对数据进行CRUD操作。

内容提供器的用法一般有两种,一种是使用现有的内容提供器来读取和操作相应程序中的数据;另一种是创建自己的内容提供器给我们的程序提供外部访问接口。

一、使用系统已有的内容提供器

android系统中自导的电话簿、短信、媒体库等程序都已有了内容提供器接口,这样第三方应用程序可以充分利用这部分数据来实现更好的功能。以读取系统联系人为例:

①查询系统联系人

  1. Cursor cursor=null;
  2. try{
  3. //查询联系人数据
  4. cursor=getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null, null, null);
  5. while(cursor.moveToNext()){
  6. //联系人姓名
  7. String contactName=cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
  8. //联系人手机号
  9. String contactNumber=cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
  10. //将得到的联系人数据存入contactsList中
  11. contactsList.add(contactName+":"+contactNumber);
  12. }
  13. }catch(Exception e){
  14. e.printStackTrace();
  15. }finally{
  16. if(cursor != null){
  17. cursor.close();
  18. }
  19. }

使用ContentResolver().query ()方法查询系统的联系人数据,第一个参数是内容URI,ContactsContract.CommonDataKinds.Phone.CONTENT_URI是系统已经定义好的用于查询联系人的URI常量。query()返回的联系人数据都存储在cursor中。然后通过对cursor遍历,将联系人姓名和手机号分别逐个读取出来存在contactList中。联系人姓名这一列对应的常量是ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME,联系人手机号这一列对应的常量是ContactsContract.CommonDataKinds.Phone.NUMBER。

②在AndroidManifest.xml中添加读取联系人的权限

  1. <uses-permission android:name="android.permission.READ_CONTACTS"/>

完整代码如下:

  1. public class MainActivity extends Activity {
  2. private ListView mLvContacts;
  3. //联系人数据
  4. List<String> contactsList=new ArrayList<String>();
  5. @Override
  6. protected void onCreate(Bundle savedInstanceState) {
  7. super.onCreate(savedInstanceState);
  8. setContentView(R.layout.activity_main);
  9. mLvContacts=(ListView) findViewById(R.id.lv_contacts);
  10. showContacts();//获得全部联系人并显示
  11. }
  12. /**
  13. * 获得全部联系人并显示
  14. */
  15. private void showContacts(){
  16. Cursor cursor=null;
  17. try{
  18. //查询联系人数据
  19. cursor=getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null, null, null);
  20. while(cursor.moveToNext()){
  21. //联系人姓名
  22. String contactName=cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
  23. //联系人手机号
  24. String contactNumber=cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
  25. //将得到的联系人数据存入contactsList中
  26. contactsList.add(contactName+":"+contactNumber);
  27. }
  28. //为listview设置adapter
  29. ArrayAdapter<String> adapter=new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,contactsList);
  30. mLvContacts.setAdapter(adapter);
  31. }catch(Exception e){
  32. e.printStackTrace();
  33. }finally{
  34. if(cursor != null){
  35. cursor.close();
  36. }
  37. }
  38. }
  39. }

二、创建自己的内容提供器

1、内容URI。使用内容提供器,需要用到内容URI,内容URI给内容提供器中的的数据建立了唯一标识符,它主要由两部分组成,权限(authority)和路径(path)。权限是用于对不同的应用程序做区分的,一般为了避免冲突,都会采用程序包名的方式进行命名。比如某个程序的包名是com.example.test,那么该程序对应的权限就可以命名为com.example.test.provider。路径则是用于对同一应用程序中不同的表做区分的,通常会添加到权限的后面。比如某个程序的数据库中存在两张表,table1和table2,这时就可以将路径分别命名为/table1和/table2,然后把权限和路径进行组合,然后加上内容URI的头部,于是内容URI最终的标准格式如下:

content://com.example.test.provider/table1

content://com.example.test.provider/table2
可以看出,内容URI非常清楚的表达出我们想要访问哪个程序中哪张表里的数据。

在得到了内容URI字符串自后,还需要将他解析成Uri对象才可以作为ContentResolver方法中的参数。解析如下:

  1. Uri uri = Uri.parse("content://com.example.test.provider/table1");

只需要调用Uri.parse()方法,就可以将内容URI字符串解析成Uri对象了。

2、新建一个类,继承ContentProvider。

  1. public class MyProvider extends ContentProvider{
  2. @Override
  3. public int delete(Uri arg0, String arg1, String[] arg2) {
  4. // TODO Auto-generated method stub
  5. return 0;
  6. }
  7. @Override
  8. public String getType(Uri arg0) {
  9. // TODO Auto-generated method stub
  10. return null;
  11. }
  12. @Override
  13. public Uri insert(Uri arg0, ContentValues arg1) {
  14. // TODO Auto-generated method stub
  15. return null;
  16. }
  17. @Override
  18. public boolean onCreate() {
  19. // TODO Auto-generated method stub
  20. return false;
  21. }
  22. @Override
  23. public Cursor query(Uri arg0, String[] arg1, String arg2, String[] arg3,
  24. String arg4) {
  25. // TODO Auto-generated method stub
  26. return null;
  27. }
  28. @Override
  29. public int update(Uri arg0, ContentValues arg1, String arg2, String[] arg3) {
  30. // TODO Auto-generated method stub
  31. return 0;
  32. }
  33. }

继承ContentProvider,需要重写六个方法:

onCreate():初始化内容提供器的时候调用。通常会在这里完成对数据库的创建和升级等操作,返回true表示内容提供器初始化成功,返回false则表示失败。注意,只有当存在ContentResolver尝试访问我们程序中的数据时,内容提供器才会被初始化。

getType():根据传入的内容URI来返回相应的MIME类型。

query():从内容提供器中查询数据。使用Uri参数来确定查询哪张表,projection参数用于确定查询哪些列,selection和selectionArgs参数用于约束查询条件,sortOrder参数用于对结果进行排序。查询的结果存放在Cursor对象中返回。

  1. Cursor cursor = getContentResolver().query(uri,projection,selection,selectionArgs,sortOrder);
  2. if(cursor != null){
  3. while(cursor.moveToNext()){
  4. String column1=cursor.getString(cursor.getColumnIndex("column1"));
  5. int column2=cursor.getInt(cursor.getColumnIndex("column2"));
  6. }
  7. cursor.close();
  8. }

insert():向内容提供器中添加一条数据。使用Uri参数来确定要添加到的表,待添加的数据保存在values参数中。添加完成后,返回一个用于表示这条新记录的URI。

  1. ContentValues values = new ContentValues();
  2. values.put("column1","text");
  3. values.put("column2",1);
  4. getContentResolver().insert(uri,values);

update():更新内容提供器中已有的数据。使用uri参数确实更新哪一张表中的数据,新数据保存在values参数中,selection和selectionArgs参数用于约束更新哪些行,受影响的行数将作为返回值返回。

  1. ContentValues values = new ContentValues();
  2. values.put("column1","");
  3. getContentResolver().update(uri,values,"column1=? and column2=?",new String[]{"text","1"});

dekete():从内容提供器中删除数据。使用uri参数确定要删除哪一张表中的数据,selection和selectionArgs参数用于约束删除哪些行,被删除的行数将作为返回值返回。

  1. getContentResolver().delete(uri,"column2=?",new String[]{"1"});

完整demo如下:

先看看结构图

Center

首先在AndroidTest项目中新建数据库类MyDatabaseHelper,代码如下:

  1. package com.example.androidtest;
  2. import android.content.Context;
  3. import android.database.sqlite.SQLiteDatabase;
  4. import android.database.sqlite.SQLiteDatabase.CursorFactory;
  5. import android.database.sqlite.SQLiteOpenHelper;
  6. public class MyDatabaseHelper extends SQLiteOpenHelper{
  7. private Context mContext;
  8. //创建表的语句
  9. public static final String CREATE_BOOK = "create table Book (id integer primary key autoincrement,author text,price real,pages integer,name text)";
  10. public MyDatabaseHelper(Context context, String name,
  11. CursorFactory factory, int version) {
  12. super(context, name, factory, version);
  13. mContext=context;
  14. }
  15. @Override
  16. public void onCreate(SQLiteDatabase db) {
  17. db.execSQL(CREATE_BOOK);//创建表
  18. }
  19. @Override
  20. public void onUpgrade(SQLiteDatabase arg0, int arg1, int arg2) {
  21. }
  22. }

然后新建MyProvider类:

  1. package com.example.androidtest;
  2. import android.content.ContentProvider;
  3. import android.content.ContentValues;
  4. import android.content.UriMatcher;
  5. import android.database.Cursor;
  6. import android.database.sqlite.SQLiteDatabase;
  7. import android.net.Uri;
  8. /**
  9. * contentProvider作为一种组件必须放在应用所在包或其子包下,主要作用是对外共享数据
  10. *
  11. */
  12. public class MyProvider extends ContentProvider{
  13. // 匹配码
  14. private static final int BOOK_DIR = 0;
  15. private static final int BOOK_ITEM = 1;
  16. public static final String AUTHORITY = "com.example.databasetest.provider";
  17. private static UriMatcher uriMatcher;
  18. private MyDatabaseHelper dbHelper;
  19. static {
  20. uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
  21. uriMatcher.addURI(AUTHORITY, "book", BOOK_DIR);//对等待匹配的URI进行匹配操作,必须符合com.example.databasetest.provider/book格式才能匹配返回BOOK_DIR,不匹配返回-1
  22. uriMatcher.addURI(AUTHORITY, "book/#", BOOK_ITEM);//#表示任意长度的数字,必须是book表中任意一行数据才能匹配返回BOOK_ITEM,不匹配返回-1
  23. }
  24. @Override
  25. public boolean onCreate() {
  26. dbHelper = new MyDatabaseHelper(this.getContext(),"BookStore.db",null,2);
  27. return true;//返回true表示初始化成功
  28. }
  29. @Override
  30. public int delete(Uri uri, String selection, String[] selectionArgs) {
  31. SQLiteDatabase db=dbHelper.getWritableDatabase();
  32. int deleteRows = 0;
  33. switch(uriMatcher.match(uri)){
  34. case BOOK_DIR:
  35. deleteRows=db.delete("Book", selection, selectionArgs);
  36. break;
  37. case BOOK_ITEM:
  38. /**
  39. * getPathSegments()方法会将内容URI权限之后的部分易“/”符号进行分割,并把分割后的结果放入到一个字符串列表中,
  40. * 那这个列表的第0个位置存放的就是路径,第一个位置存放的就是id了
  41. */
  42. String bookId=uri.getPathSegments().get(1);
  43. deleteRows=db.delete("Book", "id=?", new String[]{bookId});
  44. break;
  45. default:
  46. break;
  47. }
  48. return deleteRows;
  49. }
  50. /**
  51. * getType()方法返回MIME类型,MIME构成如下:
  52. * 必须以vnd开头;
  53. * 如果内容URI以路径结尾,则vnd后面接android.cursor.dir/,如果内容URI以id结尾,则vnd后面接android.cursor.item/;
  54. * 最后接上vnd.<权限>.<路径>.
  55. */
  56. @Override
  57. public String getType(Uri uri) {
  58. switch(uriMatcher.match(uri)){
  59. case BOOK_DIR:
  60. return "vnd.android.cursor.dir/vnd.com.example.databasetest.provider.book";
  61. case BOOK_ITEM:
  62. return "vnd.android.cursor.item/vnd.com.example.databasetest.provider.book";
  63. }
  64. return null;
  65. }
  66. @Override
  67. public Uri insert(Uri uri, ContentValues values) {
  68. SQLiteDatabase db = dbHelper.getWritableDatabase();
  69. Uri uriReturn = null;
  70. switch(uriMatcher.match(uri)){
  71. case BOOK_DIR:
  72. case BOOK_ITEM:
  73. long newBookId=db.insert("Book", null, values);
  74. uriReturn=Uri.parse("content://"+AUTHORITY+"/book/"+newBookId);
  75. break;
  76. }
  77. return uriReturn;
  78. }
  79. @Override
  80. public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
  81. String sortOrder) {
  82. SQLiteDatabase db=dbHelper.getReadableDatabase();
  83. Cursor cursor=null;
  84. switch(uriMatcher.match(uri)){
  85. case BOOK_DIR:
  86. cursor=db.query("Book", projection, selection, selectionArgs, null, null, sortOrder);
  87. break;
  88. case BOOK_ITEM:
  89. String bookId=uri.getPathSegments().get(1);
  90. cursor=db.query("Book", projection, "id=?", new String[]{bookId}, null, null, sortOrder);
  91. break;
  92. default:
  93. break;
  94. }
  95. return cursor;
  96. }
  97. @Override
  98. public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
  99. SQLiteDatabase db=dbHelper.getWritableDatabase();
  100. int updateRows=0;
  101. switch(uriMatcher.match(uri)){
  102. case BOOK_DIR:
  103. updateRows=db.update("Book", values, selection, selectionArgs);
  104. break;
  105. case BOOK_ITEM:
  106. String bookId=uri.getPathSegments().get(1);
  107. updateRows=db.update("Book", values, "id=?", new String[]{bookId});
  108. break;
  109. default:
  110. break;
  111. }
  112. return updateRows;
  113. }
  114. }

MainActivity.class类:

  1. package com.example.androidtest;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. import android.app.Activity;
  5. import android.content.BroadcastReceiver;
  6. import android.content.Context;
  7. import android.content.Intent;
  8. import android.content.IntentFilter;
  9. import android.database.Cursor;
  10. import android.net.ConnectivityManager;
  11. import android.net.NetworkInfo;
  12. import android.net.Uri;
  13. import android.os.Bundle;
  14. import android.provider.ContactsContract;
  15. import android.view.View;
  16. import android.view.View.OnClickListener;
  17. import android.widget.ArrayAdapter;
  18. import android.widget.ListView;
  19. import android.widget.Toast;
  20. public class MainActivity extends Activity {
  21. private ListView mLvBooks;
  22. //数据
  23. List<String> booksList=new ArrayList<String>();
  24. @Override
  25. protected void onCreate(Bundle savedInstanceState) {
  26. super.onCreate(savedInstanceState);
  27. setContentView(R.layout.activity_main);
  28. mLvBooks=(ListView) findViewById(R.id.lv_contacts);
  29. showContacts();//获得全部联系人并显示
  30. }
  31. /**
  32. * 获得全部联系人并显示
  33. */
  34. private void showContacts(){
  35. Cursor cursor=null;
  36. try{
  37. Uri uri=Uri.parse("content://com.example.databasetest.provider/book");
  38. //查询联系人数据
  39. cursor=getContentResolver().query(uri, null, null, null, null);
  40. if(cursor != null){
  41. while(cursor.moveToNext()){
  42. //书名
  43. String bookName=cursor.getString(cursor.getColumnIndex("name"));
  44. //作者
  45. String bookAuthor=cursor.getString(cursor.getColumnIndex("author"));
  46. //页数
  47. int pages=cursor.getInt(cursor.getColumnIndex("pages"));
  48. //价格
  49. Double price=cursor.getDouble(cursor.getColumnIndex("price"));
  50. //将得到的数据存入contactsList中
  51. booksList.add(bookName+"---作者:"+bookAuthor+" 页数:"+pages+" 价格:"+price);
  52. }
  53. }
  54. //为listview设置adapter
  55. ArrayAdapter<String> adapter=new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,booksList);
  56. mLvBooks.setAdapter(adapter);
  57. }catch(Exception e){
  58. e.printStackTrace();
  59. }finally{
  60. if(cursor != null){
  61. cursor.close();
  62. }
  63. }
  64. }
  65. }

注意:别忘了在androidManifest.xml中添加内容提供器注册代码:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.androidtest"
  4. android:versionCode="1"
  5. android:versionName="1.0" >
  6. <uses-sdk
  7. android:minSdkVersion="8"
  8. android:targetSdkVersion="19" />
  9. <application
  10. android:allowBackup="true"
  11. android:icon="@drawable/ic_launcher"
  12. android:label="@string/app_name" >
  13. <activity
  14. android:name="com.example.androidtest.MainActivity"
  15. android:label="@string/app_name" >
  16. <intent-filter>
  17. <action android:name="android.intent.action.MAIN" />
  18. <category android:name="android.intent.category.LAUNCHER" />
  19. </intent-filter>
  20. </activity>
  21. <span style="color:#cc0000;"><provider android:name="com.example.androidtest.MyProvider"
  22. android:authorities="com.example.databasetest.provider"></provider></span>
  23. </application>
  24. </manifest>

至此,AndroidTest项目已经拥有了跨程序共享数据的功能了,其他应用可以利用Uri来操作AndroidTest中数据库中的数据了。下面新建一个AndroidTest2来访问AndroidTest数据库中的数据吧!

activity.main代码:

  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. android:layout_width="fill_parent"
  3. android:layout_height="fill_parent"
  4. android:orientation="vertical" >
  5. <Button
  6. android:id="@+id/btn_add"
  7. android:layout_width="wrap_content"
  8. android:layout_height="wrap_content"
  9. android:text="add" />
  10. <Button
  11. android:id="@+id/btn_delete"
  12. android:layout_width="wrap_content"
  13. android:layout_height="wrap_content"
  14. android:text="delete" />
  15. <Button
  16. android:id="@+id/btn_update"
  17. android:layout_width="wrap_content"
  18. android:layout_height="wrap_content"
  19. android:text="update" />
  20. <Button
  21. android:id="@+id/btn_query"
  22. android:layout_width="wrap_content"
  23. android:layout_height="wrap_content"
  24. android:text="query" />
  25. <ListView
  26. android:id="@+id/lv_books"
  27. android:layout_width="match_parent"
  28. android:layout_height="wrap_content" >
  29. </ListView>
  30. </LinearLayout>

MainActivity.class:

  1. package com.example.androidtest2;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. import android.app.Activity;
  5. import android.content.ContentValues;
  6. import android.database.Cursor;
  7. import android.net.Uri;
  8. import android.os.Bundle;
  9. import android.view.View;
  10. import android.view.View.OnClickListener;
  11. import android.widget.ArrayAdapter;
  12. import android.widget.Button;
  13. import android.widget.ListView;
  14. public class MainActivity extends Activity implements OnClickListener{
  15. private ListView mLvBooks;
  16. //数据
  17. List<String> booksList=new ArrayList<String>();
  18. private Button mBtnAdd;
  19. private Button mBtnDelete;
  20. private Button mBtnUpdate;
  21. private Button mBtnQuery;
  22. String newId;//新添加的数据的id
  23. @Override
  24. protected void onCreate(Bundle savedInstanceState) {
  25. super.onCreate(savedInstanceState);
  26. setContentView(R.layout.activity_main);
  27. mLvBooks=(ListView) findViewById(R.id.lv_books);
  28. mBtnAdd=(Button) findViewById(R.id.btn_add);
  29. mBtnAdd.setOnClickListener(this);
  30. mBtnDelete=(Button) findViewById(R.id.btn_delete);
  31. mBtnDelete.setOnClickListener(this);
  32. mBtnUpdate=(Button) findViewById(R.id.btn_update);
  33. mBtnUpdate.setOnClickListener(this);
  34. mBtnQuery=(Button) findViewById(R.id.btn_query);
  35. mBtnQuery.setOnClickListener(this);
  36. }
  37. @Override
  38. public void onClick(View view) {
  39. switch(view.getId()){
  40. case R.id.btn_add:
  41. //添加
  42. Uri addUri=Uri.parse("content://com.example.databasetest.provider/book");
  43. ContentValues values=new ContentValues();
  44. values.put("name", "精彩世界");
  45. values.put("author", "郭敬明");
  46. values.put("pages", 1020);
  47. values.put("price", 75.36);
  48. Uri returnUri=getContentResolver().insert(addUri, values);
  49. newId=returnUri.getPathSegments().get(1);
  50. break;
  51. case R.id.btn_delete:
  52. //删除新增加的数据
  53. Uri deleteUri=Uri.parse("content://com.example.databasetest.provider/book/"+newId);
  54. getContentResolver().delete(deleteUri, null, null);
  55. break;
  56. case R.id.btn_update:
  57. //更新新增加的数据
  58. Uri updateUri=Uri.parse("content://com.example.databasetest.provider/book/"+newId);
  59. ContentValues updateValues=new ContentValues();
  60. updateValues.put("name", "精彩世界2");
  61. updateValues.put("price", 88.88);
  62. getContentResolver().update(updateUri, updateValues, null, null);
  63. break;
  64. case R.id.btn_query:
  65. booksList.clear();
  66. //查询
  67. Uri queryUri=Uri.parse("content://com.example.databasetest.provider/book");
  68. Cursor cursor=getContentResolver().query(queryUri, null, null, null, null);
  69. if(cursor != null){
  70. while(cursor.moveToNext()){
  71. //书名
  72. String bookName=cursor.getString(cursor.getColumnIndex("name"));
  73. //作者
  74. String bookAuthor=cursor.getString(cursor.getColumnIndex("author"));
  75. //页数
  76. int pages=cursor.getInt(cursor.getColumnIndex("pages"));
  77. //价格
  78. Double price=cursor.getDouble(cursor.getColumnIndex("price"));
  79. //将得到的数据存入contactsList中
  80. booksList.add(bookName+"---作者:"+bookAuthor+" 页数:"+pages+" 价格:"+price);
  81. }
  82. //为listview设置adapter
  83. ArrayAdapter<String> adapter=new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,booksList);
  84. mLvBooks.setAdapter(adapter);
  85. }
  86. break;
  87. default:
  88. break;
  89. }
  90. }
  91. }

我们在AndroidTest2的四个按钮的点击事件中处理了AndroidTest数据库中增删改查的操作。
可以看出,我们的跨程序共享数据功能已经实现了!现在不仅仅是AndroidTest2程序,任何一个程序都可以轻松访问AndroidTest中的数据,而且不存在隐私数据泄露的问题。

发表评论

表情:
评论列表 (有 0 条评论,278人围观)

还没有评论,来说两句吧...

相关阅读