创建及调用自己的ContentProvider

转自:http://blog.csdn.net/liuhe688/article/details/7050868

===================================================================

今天我们来讲解一下如何创建及调用自己的ContentProvider。

在前面两篇文章中我们分别讲了如何读写联系人和短消息,相信大家对于ContentProvider的操作方法已经有了一定程度的了解在有些场合,除了操作ContentProvider之外,我们还有可能需要创建自己的ContentProvider,来提供信息共享的服务,这就要求我们很好的掌握ContentProvider的创建及使用技巧下面我们就由表及里的逐步讲解每个步骤

在正式开始实例演示之前,我们先来了解以下两个知识点:

在Android中,每一个ContentProvider都会用类似于域名的字符串来注册自己,我们成为授权(authority)这个唯一标识的字符串是此ContentProvider可提供的一组URI的基础,有了这个基础,才能够向外界提供信息的共享服务

授权是在AndroidManifest.xml中完成的,每一个ContentProvider必须在此声明并授权,方式如下:

  1. <provider android:name=".SomeProvider"  
  2.     android:authorities="com.your-company.SomeProvider"/>  
		<provider android:name=".SomeProvider"
			android:authorities="com.your-company.SomeProvider"/>
上面的<provider>元素指明了ContentProvider的提供者是“SomeProvider”这个类,并为其授权,授权的基础URI为“com.your-company.SomeProvider”有了这个授权信息,系统可以准确的定位到具体的ContentProvider,从而使访问者能够获取到指定的信息这和浏览Web页面的方式很相似,“SomeProvider”就像一台具体的服务器,而“com.your-company.SomeProvider”就像注册的域名,相信大家对这个概念并不陌生,由此联想一下就可以了解ContentProvider授权的作用了(需要注意的是,除了Android内置应用程序之外,第三方程序应尽量使用以上方式的完全限定的授权名

MIME类型:

就像网站返回给定URL的MIME(Multipurpose Internet Mail Extensions,多用途Internet邮件扩展)类型一样(这使浏览器能够用正确的程序来查看内容),ContentProvider还负责返回给定URI的MIME类型根据MIME类型规范,MIME类型包含两部分:类型和子类型例如:text/html,text/css,text/xml等等。

Android也遵循类似的约定来定义MIME类型

对于单条记录,MIME类型类似于:

vnd.android.cursor.item/vnd.your-company.content-type

而对于记录的集合,MIME类型类似于:

vnd.android.cursor.dir/vnd.your-company.comtent-type

其中的vnd表示这些类型和子类型具有非标准的供应商特定的形式;content-type可以根据ContentProvider的功能来定,比如日记的ContentProvider可以为note,日程安排的ContentProvider可以为schedule,等等

了解了以上两个知识点之后,我们就结合实例来演示一下具体的过程。

我们将会创建一个记录person信息的ContentProvider,实现对person的CRUD操作,访问者可以通过下面路径操作我们的ContentProvider:

 

 

访问者可以通过“[BASE_URI]/persons”来操作person集合,也可以通过“[BASE_URI]/persons/#”的形式操作单个person

我们创建一个person的ContentProvider需要两个步骤:

1.创建PersonProvider类:

我们需要继承ContentProvider类,实现onCreatequery、insert、update、delete和getType这几个方法具体代码如下:

  1. package com.scott.provider;  
  2. import android.content.ContentProvider;  
  3. import android.content.ContentUris;  
  4. import android.content.ContentValues;  
  5. import android.content.UriMatcher;  
  6. import android.database.Cursor;  
  7. import android.database.sqlite.SQLiteDatabase;  
  8. import android.net.Uri;  
  9. public class PersonProvider extends ContentProvider {  
  10.     private static final UriMatcher matcher;  
  11.     private DBHelper helper;  
  12.     private SQLiteDatabase db;  
  13.     private static final String AUTHORITY = "com.scott.provider.PersonProvider";  
  14.     private static final int PERSON_ALL = 0;  
  15.     private static final int PERSON_ONE = 1;  
  16.     public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.scott.person";  
  17.     public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.scott.person";  
  18.     //数据改变后立即重新查询  
  19.     private static final Uri NOTIFY_URI = Uri.parse("content://" + AUTHORITY + "/persons");  
  20.     static {  
  21.         matcher = new UriMatcher(UriMatcher.NO_MATCH);  
  22.         matcher.addURI(AUTHORITY, "persons", PERSON_ALL);   //匹配记录集合  
  23.         matcher.addURI(AUTHORITY, "persons/#", PERSON_ONE); //匹配单条记录  
  24.     @Override  
  25.     public boolean onCreate() {  
  26.         helper = new DBHelper(getContext());  
  27.         return true;  
  28.     @Override  
  29.     public String getType(Uri uri) {  
  30.         int match = matcher.match(uri);  
  31.         switch (match) {  
  32.         case PERSON_ALL:  
  33.             return CONTENT_TYPE;  
  34.         case PERSON_ONE:  
  35.             return CONTENT_ITEM_TYPE;  
  36.         default:  
  37.             throw new IllegalArgumentException("Unknown URI: " + uri);  
  38.     @Override  
  39.     public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {  
  40.         db = helper.getReadableDatabase();  
  41.         int match = matcher.match(uri);  
  42.         switch (match) {  
  43.         case PERSON_ALL:  
  44.             //doesn't need any code in my provider.  
  45.             break;  
  46.         case PERSON_ONE:  
  47.             long _id = ContentUris.parseId(uri);  
  48.             selection = "_id = ?";  
  49.             selectionArgs = new String[]{String.valueOf(_id)};  
  50.             break;  
  51.         default:  
  52.             throw new IllegalArgumentException("Unknown URI: " + uri);  
  53.         return db.query("person", projection, selection, selectionArgs, null, null, sortOrder);  
  54.     @Override  
  55.     public Uri insert(Uri uri, ContentValues values) {  
  56.         int match = matcher.match(uri);  
  57.         if (match != PERSON_ALL) {  
  58.             throw new IllegalArgumentException("Wrong URI: " + uri);  
  59.         db = helper.getWritableDatabase();  
  60.         if (values == null) {  
  61.             values = new ContentValues();  
  62.             values.put("name", "no name");  
  63.             values.put("age", "1");  
  64.             values.put("info", "no info.");  
  65.         long rowId = db.insert("person", null, values);  
  66.         if (rowId > 0) {  
  67.             notifyDataChanged();  
  68.             return ContentUris.withAppendedId(uri, rowId);  
  69.         return null;  
  70.     @Override  
  71.     public int delete(Uri uri, String selection, String[] selectionArgs) {  
  72.         db = helper.getWritableDatabase();  
  73.         int match = matcher.match(uri);  
  74.         switch (match) {  
  75.         case PERSON_ALL:  
  76.             //doesn't need any code in my provider.  
  77.             break;  
  78.         case PERSON_ONE:  
  79.             long _id = ContentUris.parseId(uri);  
  80.             selection = "_id = ?";  
  81.             selectionArgs = new String[]{String.valueOf(_id)};  
  82.         int count = db.delete("person", selection, selectionArgs);  
  83.         if (count > 0) {  
  84.             notifyDataChanged();  
  85.         return count;  
  86.     @Override  
  87.     public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {  
  88.         db = helper.getWritableDatabase();  
  89.         int match = matcher.match(uri);  
  90.         switch (match) {  
  91.         case PERSON_ALL:  
  92.             //doesn't need any code in my provider.  
  93.             break;  
  94.         case PERSON_ONE:  
  95.             long _id = ContentUris.parseId(uri);  
  96.             selection = "_id = ?";  
  97.             selectionArgs = new String[]{String.valueOf(_id)};  
  98.             break;  
  99.         default:  
  100.             throw new IllegalArgumentException("Unknown URI: " + uri);  
  101.         int count = db.update("person", values, selection, selectionArgs);  
  102.         if (count > 0) {  
  103.             notifyDataChanged();  
  104.         return count;  
  105.     //通知指定URI数据已改变  
  106.     private void notifyDataChanged() {  
  107.         getContext().getContentResolver().notifyChange(NOTIFY_URI, null);         
package com.scott.provider;

import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.net.Uri;

public class PersonProvider extends ContentProvider {

	private static final UriMatcher matcher;
	private DBHelper helper;
	private SQLiteDatabase db;
	
	private static final String AUTHORITY = "com.scott.provider.PersonProvider";
	private static final int PERSON_ALL = 0;
	private static final int PERSON_ONE = 1;
	
	public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.scott.person";
	public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.scott.person";
	
	//数据改变后立即重新查询
	private static final Uri NOTIFY_URI = Uri.parse("content://" + AUTHORITY + "/persons");
	
	static {
		matcher = new UriMatcher(UriMatcher.NO_MATCH);
		
		matcher.addURI(AUTHORITY, "persons", PERSON_ALL);	//匹配记录集合
		matcher.addURI(AUTHORITY, "persons/#", PERSON_ONE);	//匹配单条记录
	}
	
	@Override
	public boolean onCreate() {
		helper = new DBHelper(getContext());
		return true;
	}

	@Override
	public String getType(Uri uri) {
		int match = matcher.match(uri);
		switch (match) {
		case PERSON_ALL:
			return CONTENT_TYPE;
		case PERSON_ONE:
			return CONTENT_ITEM_TYPE;
		default:
			throw new IllegalArgumentException("Unknown URI: " + uri);
		}
	}
	
	@Override
	public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
		db = helper.getReadableDatabase();
		int match = matcher.match(uri);
		switch (match) {
		case PERSON_ALL:
			//doesn't need any code in my provider.
			break;
		case PERSON_ONE:
			long _id = ContentUris.parseId(uri);
			selection = "_id = ?";
			selectionArgs = new String[]{String.valueOf(_id)};
			break;
		default:
			throw new IllegalArgumentException("Unknown URI: " + uri);
		}
		return db.query("person", projection, selection, selectionArgs, null, null, sortOrder);
	}

	@Override
	public Uri insert(Uri uri, ContentValues values) {
		int match = matcher.match(uri);
		if (match != PERSON_ALL) {
			throw new IllegalArgumentException("Wrong URI: " + uri);
		}
		db = helper.getWritableDatabase();
		if (values == null) {
			values = new ContentValues();
			values.put("name", "no name");
			values.put("age", "1");
			values.put("info", "no info.");
		}
		long rowId = db.insert("person", null, values);
		if (rowId > 0) {
			notifyDataChanged();
			return ContentUris.withAppendedId(uri, rowId);
		}
		return null;
	}

	@Override
	public int delete(Uri uri, String selection, String[] selectionArgs) {
		db = helper.getWritableDatabase();
		int match = matcher.match(uri);
		switch (match) {
		case PERSON_ALL:
			//doesn't need any code in my provider.
			break;
		case PERSON_ONE:
			long _id = ContentUris.parseId(uri);
			selection = "_id = ?";
			selectionArgs = new String[]{String.valueOf(_id)};
		}
		int count = db.delete("person", selection, selectionArgs);
		if (count > 0) {
			notifyDataChanged();
		}
		return count;
	}

	@Override
	public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
		db = helper.getWritableDatabase();
		int match = matcher.match(uri);
		switch (match) {
		case PERSON_ALL:
			//doesn't need any code in my provider.
			break;
		case PERSON_ONE:
			long _id = ContentUris.parseId(uri);
			selection = "_id = ?";
			selectionArgs = new String[]{String.valueOf(_id)};
			break;
		default:
			throw new IllegalArgumentException("Unknown URI: " + uri);
		}
		int count = db.update("person", values, selection, selectionArgs);
		if (count > 0) {
			notifyDataChanged();
		}
		return count;
	}

	//通知指定URI数据已改变
	private void notifyDataChanged() {
		getContext().getContentResolver().notifyChange(NOTIFY_URI, null);		
	}
}
在PersonProvider中,我们定义了授权地址为“com.scott.provider.PersonProvider”,相信大家在前面也有所了解了基于这个授权,我们使用了一个UriMatcher对其路径进行匹配,“[BASE_URI]/persons"和“[BASE_URI]/persons/#”这两种路径我们在上面也介绍过,分别对应记录集合和单个记录的操作在query、insert、update和delete方法中我们根据UriMatcher匹配结果来判断该URI是操作记录集合还是单条记录,从而采取不同的处理方法在getType方法中,我们会根据匹配的结果返回不同的MIME类型,这一步是不能缺少的,比如我们在query方法中有可能是查询全部集合,有可能是查询单条记录,那么我们返回的Cursor或是集合类型,或是单条记录,这个跟getType返回的MIME类型是一致的,就好像浏览网页一样,指定的url返回的信息是什么类型,那么浏览器就应该接收到对应的MIME类型另外,我们注意到,上面代码中,在insert、updatedelete方法中都调用了notifyDataChanged方法,这个方法中仅有的一步操作就是通知“[BASE_URI]/persons"的访问者,数据发生改变了,应该重新加载了

在我们的PersonProvider中,我们用到了PersonDBHelper类,代码如下:

  1. package com.scott.provider;  
  2. public class Person {  
  3.     public int _id;  
  4.     public String name;  
  5.     public int age;  
  6.     public String info;  
  7.     public Person() {  
  8.     public Person(String name, int age, String info) {  
  9.         this.name = name;  
  10.         this.age = age;  
  11.         this.info = info;  
package com.scott.provider;

public class Person {
	public int _id;
	public String name;
	public int age;
	public String info;
	
	public Person() {
	}
	
	public Person(String name, int age, String info) {
		this.name = name;
		this.age = age;
		this.info = info;
	}
}
  1. package com.scott.provider;  
  2. import android.content.Context;  
  3. import android.database.sqlite.SQLiteDatabase;  
  4. import android.database.sqlite.SQLiteOpenHelper;  
  5. public class DBHelper extends SQLiteOpenHelper {  
  6.     private static final String DATABASE_NAME = "provider.db";  
  7.     private static final int DATABASE_VERSION = 1;  
  8.     public DBHelper(Context context) {  
  9.         super(context, DATABASE_NAME, null, DATABASE_VERSION);  
  10.     @Override  
  11.     public void onCreate(SQLiteDatabase db) {  
  12.         String sql = "CREATE TABLE IF NOT EXISTS person" +  
  13.                 "(_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age INTEGER, info TEXT)";  
  14.         db.execSQL(sql);  
  15.     @Override  
  16.     public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {  
  17.         db.execSQL("DROP TABLE IF EXISTS person");  
  18.         onCreate(db);  
package com.scott.provider;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;

public class DBHelper extends SQLiteOpenHelper {

	private static final String DATABASE_NAME = "provider.db";
	private static final int DATABASE_VERSION = 1;
	
	public DBHelper(Context context) {
		super(context, DATABASE_NAME, null, DATABASE_VERSION);
	}

	@Override
	public void onCreate(SQLiteDatabase db) {
		String sql = "CREATE TABLE IF NOT EXISTS person" +
				"(_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age INTEGER, info TEXT)";
		db.execSQL(sql);
	}

	@Override
	public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
		db.execSQL("DROP TABLE IF EXISTS person");
		onCreate(db);
	}
}

最后,要想让这个ContentProvider生效,我们需要在AndroidManifest.xml中声明并为其授权,如下所示:

  1. <provider android:name=".PersonProvider"  
  2.     android:authorities="com.scott.provider.PersonProvider"  
  3.     android:multiprocess="true"/>  
		<provider android:name=".PersonProvider"
			android:authorities="com.scott.provider.PersonProvider"
			android:multiprocess="true"/>

其中,android:multiprocess代表是否允许多进程操作另外我们也可以为其声明相应的权限,对应的属性是:android:permission

2.调用PersonProvider类:

完成了person的ContentProvider后,下面我们来看一下如何访问它这一步我们在MainActivity中完成,看下面代码:

  1. package com.scott.provider;  
  2. import java.util.ArrayList;  
  3. import android.app.Activity;  
  4. import android.content.ContentResolver;  
  5. import android.content.ContentUris;  
  6. import android.content.ContentValues;  
  7. import android.database.Cursor;  
  8. import android.database.CursorWrapper;  
  9. import android.net.Uri;  
  10. import android.os.Bundle;  
  11. import android.os.Handler;  
  12. import android.os.Message;  
  13. import android.view.View;  
  14. import android.widget.ListView;  
  15. import android.widget.SimpleCursorAdapter;  
  16. public class MainActivity extends Activity {  
  17.     private ContentResolver resolver;  
  18.     private ListView listView;  
  19.     private static final String AUTHORITY = "com.scott.provider.PersonProvider";  
  20.     private static final Uri PERSON_ALL_URI = Uri.parse("content://" + AUTHORITY + "/persons");  
  21.     private Handler handler = new Handler() {  
  22.         public void handleMessage(Message msg) {  
  23.             //update records.  
  24.             requery();  
  25.     @Override  
  26.     public void onCreate(Bundle savedInstanceState) {  
  27.         super.onCreate(savedInstanceState);  
  28.         setContentView(R.layout.main);  
  29.         resolver = getContentResolver();  
  30.         listView = (ListView) findViewById(R.id.listView);  
  31.         //为PERSON_ALL_URI注册变化通知  
  32.         getContentResolver().registerContentObserver(PERSON_ALL_URI, true, new PersonObserver(handler));  
  33.      * @param view 
  34.     public void init(View view) {  
  35.         ArrayList<Person> persons = new ArrayList<Person>();  
  36.         Person person1 = new Person("Ella", 22, "lively girl");  
  37.         Person person2 = new Person("Jenny", 22, "beautiful girl");  
  38.         Person person3 = new Person("Jessica", 23, "sexy girl");  
  39.         Person person4 = new Person("Kelly", 23, "hot baby");  
  40.         Person person5 = new Person("Jane", 25, "pretty woman");  
  41.         persons.add(person1);  
  42.         persons.add(person2);  
  43.         persons.add(person3);  
  44.         persons.add(person4);  
  45.         persons.add(person5);  
  46.         for (Person person : persons) {  
  47.             ContentValues values = new ContentValues();  
  48.             values.put("name", person.name);  
  49.             values.put("age", person.age);  
  50.             values.put("info", person.info);  
  51.             resolver.insert(PERSON_ALL_URI, values);  
  52.      * 查询所有记录 
  53.      * @param view 
  54.     public void query(View view) {  
  55. //      Uri personOneUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);查询_id为1的记录  
  56.         Cursor c = resolver.query(PERSON_ALL_URI, null, null, null, null);  
  57.         CursorWrapper cursorWrapper = new CursorWrapper(c) {  
  58.             @Override  
  59.             public String getString(int columnIndex) {  
  60.                 //将简介前加上年龄  
  61.                 if (getColumnName(columnIndex).equals("info")) {  
  62.                     int age = getInt(getColumnIndex("age"));  
  63.                     return age + " years old, " + super.getString(columnIndex);  
  64.                 return super.getString(columnIndex);  
  65.         //Cursor须含有"_id"字段  
  66.         SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, android.R.layout.simple_list_item_2,  
  67.                 cursorWrapper, new String[]{"name", "info"}, new int[]{android.R.id.text1, android.R.id.text2});  
  68.         listView.setAdapter(adapter);  
  69.         startManagingCursor(cursorWrapper); //管理Cursor  
  70.      * 插入一条记录 
  71.      * @param view 
  72.     public void insert(View view) {  
  73.         Person person = new Person("Alina", 26, "attractive lady");  
  74.         ContentValues values = new ContentValues();  
  75.         values.put("name", person.name);  
  76.         values.put("age", person.age);  
  77.         values.put("info", person.info);  
  78.         resolver.insert(PERSON_ALL_URI, values);  
  79.      * 更新一条记录 
  80.      * @param view 
  81.     public void update(View view) {  
  82.         Person person = new Person();  
  83.         person.name = "Jane";  
  84.         person.age = 30;  
  85.         //将指定name的记录age字段更新为30  
  86.         ContentValues values = new ContentValues();  
  87.         values.put("age", person.age);  
  88.         resolver.update(PERSON_ALL_URI, values, "name = ?", new String[]{person.name});  
  89.         //将_id为1的age更新为30  
  90. //      Uri updateUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);  
  91. //      resolver.update(updateUri, values, null, null);  
  92.      * 删除一条记录 
  93.      * @param view 
  94.     public void delete(View view) {  
  95.         //删除_id为1的记录  
  96.         Uri delUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);  
  97.         resolver.delete(delUri, null, null);  
  98.         //删除所有记录  
  99. //      resolver.delete(PERSON_ALL_URI, null, null);  
  100.      * 重新查询 
  101.     private void requery() {  
  102.         //实际操作中可以查询集合信息后Adapter.notifyDataSetChanged();  
  103.         query(null);  
package com.scott.provider;

import java.util.ArrayList;

import android.app.Activity;
import android.content.ContentResolver;
import android.content.ContentUris;
import android.content.ContentValues;
import android.database.Cursor;
import android.database.CursorWrapper;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.ListView;
import android.widget.SimpleCursorAdapter;


public class MainActivity extends Activity {
   
	private ContentResolver resolver;
	private ListView listView;
	
	private static final String AUTHORITY = "com.scott.provider.PersonProvider";
	private static final Uri PERSON_ALL_URI = Uri.parse("content://" + AUTHORITY + "/persons");
	
	private Handler handler = new Handler() {
		public void handleMessage(Message msg) {
			//update records.
			requery();
		};
	};
	
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        resolver = getContentResolver();
        listView = (ListView) findViewById(R.id.listView);
        
        //为PERSON_ALL_URI注册变化通知
        getContentResolver().registerContentObserver(PERSON_ALL_URI, true, new PersonObserver(handler));
    }
    
    /**
     * 初始化
     * @param view
     */
    public void init(View view) {
    	ArrayList<Person> persons = new ArrayList<Person>();
    	
    	Person person1 = new Person("Ella", 22, "lively girl");
    	Person person2 = new Person("Jenny", 22, "beautiful girl");
    	Person person3 = new Person("Jessica", 23, "sexy girl");
    	Person person4 = new Person("Kelly", 23, "hot baby");
    	Person person5 = new Person("Jane", 25, "pretty woman");
    	
    	persons.add(person1);
    	persons.add(person2);
    	persons.add(person3);
    	persons.add(person4);
    	persons.add(person5);

    	for (Person person : persons) {
    		ContentValues values = new ContentValues();
    		values.put("name", person.name);
    		values.put("age", person.age);
    		values.put("info", person.info);
    		resolver.insert(PERSON_ALL_URI, values);
    	}
    }
    
    /**
     * 查询所有记录
     * @param view
     */
    public void query(View view) {
//    	Uri personOneUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);查询_id为1的记录
    	Cursor c = resolver.query(PERSON_ALL_URI, null, null, null, null);
    	
    	CursorWrapper cursorWrapper = new CursorWrapper(c) {
    		
    		@Override
    		public String getString(int columnIndex) {
    			//将简介前加上年龄
    			if (getColumnName(columnIndex).equals("info")) {
    				int age = getInt(getColumnIndex("age"));
    				return age + " years old, " + super.getString(columnIndex);
    			}
    			return super.getString(columnIndex);
    		}
    	};
    	
    	//Cursor须含有"_id"字段
    	SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, android.R.layout.simple_list_item_2,
    			cursorWrapper, new String[]{"name", "info"}, new int[]{android.R.id.text1, android.R.id.text2});
    	listView.setAdapter(adapter);
    	
    	startManagingCursor(cursorWrapper);	//管理Cursor
    }
    
    /**
     * 插入一条记录
     * @param view
     */
    public void insert(View view) {
    	Person person = new Person("Alina", 26, "attractive lady");
    	ContentValues values = new ContentValues();
		values.put("name", person.name);
		values.put("age", person.age);
		values.put("info", person.info);
		resolver.insert(PERSON_ALL_URI, values);
    }
    
    /**
     * 更新一条记录
     * @param view
     */
    public void update(View view) {
    	Person person = new Person();
    	person.name = "Jane";
    	person.age = 30;
    	//将指定name的记录age字段更新为30
    	ContentValues values = new ContentValues();
    	values.put("age", person.age);
    	resolver.update(PERSON_ALL_URI, values, "name = ?", new String[]{person.name});
    	
    	//将_id为1的age更新为30
//    	Uri updateUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);
//    	resolver.update(updateUri, values, null, null);
    }
    
    /**
     * 删除一条记录
     * @param view
     */
    public void delete(View view) {
    	//删除_id为1的记录
    	Uri delUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);
    	resolver.delete(delUri, null, null);
    	
    	//删除所有记录
//    	resolver.delete(PERSON_ALL_URI, null, null);
    }
    
    /**
     * 重新查询
     */
    private void requery() {
    	//实际操作中可以查询集合信息后Adapter.notifyDataSetChanged();
    	query(null);
    }
}
我们看到,在上面的代码中,分别对应每一种情况进行测试,相对较为简单我们主要讲一下registerContentObserver这一环节

在前面的PersonProvider我们也提到,在数据更改后,会向指定的URI访问者发出通知,以便于更新查询记录大家注意,仅仅是ContentProvider出力还不够,我们还需要在访问者中注册一个ContentObserver,才能够接收到这个通知下面我们创建一个PersonObserver:

  1. package com.scott.provider;  
  2. import android.database.ContentObserver;  
  3. import android.os.Handler;  
  4. import android.os.Message;  
  5. import android.util.Log;  
  6. public class PersonObserver extends ContentObserver {  
  7.     public static final String TAG = "PersonObserver";  
  8.     private Handler handler;  
  9.     public PersonObserver(Handler handler) {  
  10.         super(handler);  
  11.         this.handler = handler;  
  12.     @Override  
  13.     public void onChange(boolean selfChange) {  
  14.         super.onChange(selfChange);  
  15.         Log.i(TAG, "data changed, try to requery.");  
  16.         //向handler发送消息,更新查询记录  
  17.         Message msg = new Message();  
  18.         handler.sendMessage(msg);  
package com.scott.provider;

import android.database.ContentObserver;
import android.os.Handler;
import android.os.Message;
import android.util.Log;

public class PersonObserver extends ContentObserver {

	public static final String TAG = "PersonObserver";
	private Handler handler;
	
	public PersonObserver(Handler handler) {
		super(handler);
		this.handler = handler;
	}
	
	@Override
	public void onChange(boolean selfChange) {
		super.onChange(selfChange);
		Log.i(TAG, "data changed, try to requery.");
		//向handler发送消息,更新查询记录
		Message msg = new Message();
		handler.sendMessage(msg);
	}
}
这样一来,当ContentProvider发来通知之后,我们就能立即接收到,从而向handler发送一条消息,重新查询记录,使我们能够看到最新的记录信息

最后,我们要在AndroidManifest.xml中为MainActivity添加MIME类型过滤器,告诉系统MainActivity可以处理的信息类型:

  1. <!-- MIME类型 -->  
  2. <intent-filter>  
  3.     <data android:mimeType="vnd.android.cursor.dir/vnd.scott.person"/>  
  4. </intent-filter>  
  5. <intent-filter>  
  6.     <data android:mimeType="vnd.android.cursor.item/vnd.scott.person"/>  
  7. </intent-filter>  
            <!-- MIME类型 -->
            <intent-filter>
            	<data android:mimeType="vnd.android.cursor.dir/vnd.scott.person"/>
            </intent-filter>
            <intent-filter>
            	<data android:mimeType="vnd.android.cursor.item/vnd.scott.person"/>
            </intent-filter>
这样就完成了访问者的代码,我们来看一下效果:

 

 

鉴于操作类型太多,我在这里就不再展示了,大家可以自己试一试


 
 
 
 
 
posted @ 2013-02-15 16:30  __木头鱼__  阅读(240)  评论(0编辑  收藏  举报