写在转帖前:最近遇到个问题,使用ContentResolver插入数据然后取出该条数据,当时是正常的,但当把进程杀掉后,下次启动程序时,程序竟然重新调用ContentProvider新建数据库,导致数据丢失。以下仅是明确两者使用方法并不是问题答案。
android中对数据操作包含有:
file, sqlite3, Preferences, ContectResolver与ContentProvider前三种数据操作方式都只是针对本应用内数据,程序不能通过这三种方法去操作别的应用内的数据。
android中提供ContectResolver与ContentProvider来操作别的应用程序的数据。
一、 使用方式
一个应用实现ContentProvider来提供内容给别的应用来操作,
一个应用通过ContentResolver来操作别的应用数据,当然在自己的应用中也可以。
1. ContentResolver的获取
通过Context类:
- public abstract ContentResolver getContentResolver();
public abstract ContentResolver getContentResolver();
2. ContentResolver常用操作
-
- public final Cursor query(Uri uri, String[] projection,
- String selection, String[] selectionArgs, String sortOrder);
-
- public final Uri insert(Uri url, ContentValues values)
-
- public final int update(Uri uri, ContentValues values, String where,
- String[] selectionArgs)
-
- public final int delete(Uri url, String where, String[] selectionArgs)
-
//查询:
public final Cursor query(Uri uri, String[] projection,
String selection, String[] selectionArgs, String sortOrder);
//新增
public final Uri insert(Uri url, ContentValues values)
//更新
public final int update(Uri uri, ContentValues values, String where,
String[] selectionArgs)
//删除
public final int delete(Uri url, String where, String[] selectionArgs)
以上操作实际是通过Uri来匹配ContentProvider, 再由ContentProvider来进行具体操作的。
操作的参数和操作sqlite各函数的参数意义是一样的。
二、实现ContentProvider提供给外界访问
调用者ContentResoler是通过一个Uri来找到相应的ContentProvider的来进行实际操作。
1. Uri概念
一个Uri的样子如:
scheme://authorities/path/id
如电话记录:
- public static final Uri CONTENT_URI = Uri.parse("content://call_log/calls");
public static final Uri CONTENT_URI = Uri.parse("content://call_log/calls");
a.根据scheme不同调用不程序来处理, 常用的:content, android_resource, file, http等
b.authorities是provider定义的,在AndroidManifest.xml中定义
c.path和id就好理解的。
2. Uri定义
创建自己的Uri, 如:
content://com.shguo.statistic/sms
一般数据中都有dir和item两种(当然可定义多个)。为ContentProvider创建息的UriMatcher并添加这两者:
- String AUTHORITY = "com.shguo.statistics";
- UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
- sUriMatcher.addURI(AUTHORITY, "sms", SMS_DIR);
- sUriMatcher.addURI(AUTHORITY, "sms/#", SMS_ITEM);
String AUTHORITY = "com.shguo.statistics";
UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
sUriMatcher.addURI(AUTHORITY, "sms", SMS_DIR); //SMS_DIR = 1
sUriMatcher.addURI(AUTHORITY, "sms/#", SMS_ITEM); //SMS_ITEM = 2
contentProvider要根据传入uri判断是dir还是item来操作的。
- switch (sUriMatcher.match(uri))
switch (sUriMatcher.match(uri))
来分步操作.
3. 定义MIME类型,
覆盖getType方法:主要是根据uri来返回Provider的MIME类型
- public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.shguo.sms";
- ublic static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.shguo.sms";
public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.shguo.sms";
public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.shguo.sms";
getType()为:
- switch (sUriMatcher.match(uri)) {
- case SMS_DIR:
- return CONTENT_TYPE;
- case SMS_ITEM:
- return CONTENT_ITEM_TYPE;
- default:
- throw new IllegalArgumentException("Unknown URI " + uri);
- }
switch (sUriMatcher.match(uri)) {
case SMS_DIR:
return CONTENT_TYPE;
case SMS_ITEM:
return CONTENT_ITEM_TYPE;
default:
throw new IllegalArgumentException("Unknown URI " + uri);
}
4. 实现query, insert, delete, update四个操作。
具体的实现可以用sqlite, file等。并根据uri分情况操作。
a. query时如果是item加查询条件id.
where = "_ID=" + uri.getPathSegments().get(1) + (!TextUtils.isEmpty(where) ? " AND (" + where + ')' : "";
最后要加上
cursor.setNotificationUri(getContext().getContentResolver(), uri);
b. insert时要求uri只能是dir. 成功之后返回一个加id的uri.
- Uri insertUri = ContentUris.withAppendedId(CONTENT_URI, rowId);
Uri insertUri = ContentUris.withAppendedId(CONTENT_URI, rowId);
c. update、delete与query差不多。
-
- getContext().getContentResolver().notifyChange(uri, null);
//注意通知注册uri的观察者。
getContext().getContentResolver().notifyChange(uri, null);
5. 在AndroidManifest.xml中定义
provider元素,主要属性有:
- name => ContentProvider类名
- authorities => content type的授权部分
- multiprocess => true允许在每个客户进程中创建provider实例,消除执行IPC的需求。
----------------------------------------------------------------------------------------------------------------------------------------------------
- android有一个独特之处就是,数据库只能被它的创建者所使用,其他的应用是不能访问到的,所以如果你想实现不同应用之间的数据共享,就不得不用content provider了。
- 在Android中,content provider是一个特殊的存储数据的类型,它提供了一套标准的接口用来获取以及操作数据。并且,android自身也提供了几个现成的content provider:Contacts, Browser, CallLog, Settings, MediaStore.
-
- 应用可以通过一个唯一的ContentResolver interface来使用具体的某个content provider。
-
- ContentResolver cr = getContentResolver();
- 然后你就可以用ContentResolver提供的方法来使用你需要的content provider了。其中contentResolver提供的方法包括query(),insert(),update()等。要使用这些方法,还会涉及到一个东西,那就是Uri。你可以将它理解成一个string形式的contentProvider的完全路径,它的形式为<standard_prefix>:
- 例如:
- content:
- content:
- content:
- 下面结合一个实例来看我们如何使用一个已有的content provider,给例子展示了如何从已有的电话本中读出联系人信息:
-
- package com.android.cp;
-
- import android.app.Activity;
- import android.content.ContentResolver;
- import android.database.Cursor;
- import android.os.Bundle;
- import android.provider.Contacts.People;
- import android.util.Log;
- import android.widget.Toast;
-
- public class ContentProviderTest extends Activity {
- private final String TAG = "ContentProviderTest";
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- Log.i(TAG,"enter onCreate");
- setContentView(R.layout.main);
- createCP();
- }
-
- public void createCP()
- {
- ContentResolver cr = getContentResolver();
-
-
-
- Cursor cur = cr.query(People.CONTENT_URI, null, null, null, null);
-
- getColumnData(cur);
-
- }
-
- private void getColumnData(Cursor cur){
- if (cur.moveToFirst()) {
-
- String name;
- String phoneNumber;
- int nameColumn = cur.getColumnIndex(People.NAME);
- int phoneColumn = cur.getColumnIndex(People.NUMBER);
-
- do {
-
-
- name = cur.getString(nameColumn);
- phoneNumber = cur.getString(phoneColumn);
-
- Log.i(TAG, "name="+name);
- DisplayToast(name+" "+phoneNumber);
-
- } while (cur.moveToNext());
-
- }
- }
-
- public void DisplayToast(String s)
- {
- Toast.makeText(this,
- s,
- Toast.LENGTH_LONG).show();
- }
-
- }
- 需要注意的是,你需要在你的Manifest文件中加上
-
- <uses-permission
- android:name="android.permission.READ_CONTACTS">
- </uses-permission>
- 否则,程序无法成功运行。