Android的四大元件之一,ContentProvider,分為兩個部分:一是資料組織工具ContentProvider,一是資料調用工具ContentResolver。對于系統已經提供的URI,如通訊錄、多媒體、短信等,通常可以不需要再建立額外的資料庫,再用ContentProvider進行組織,而是可以直接用ContentResolver調用這些URI,對系統資料庫進行增删改查等操作,進而保證整個Android裝置中資料的統一。
下面這個連結是本人前段時間為了參加MM論壇的一個小比賽所做的通訊錄小demo的源碼。因為最近工作和生活的一些原因,把比賽的截止日期給耽誤了,是以到最後也沒做美工。
如下是對通訊錄表進行添加和删除的處理類ContactHandler:
[java] view plaincopy
- import android.content.ContentResolver;
- import android.content.ContentUris;
- import android.content.ContentValues;
- import android.content.Context;
- import android.provider.ContactsContract.Data;
- import android.provider.ContactsContract.RawContacts;
- import android.provider.ContactsContract.CommonDataKinds.Phone;
- import android.provider.ContactsContract.CommonDataKinds.StructuredName;
- public class ContactHandler {
- private Context context;
- public ContactHandler(Context context) {
- this.context = context;
- }
- public void create(ContactItem item) {
- ContentResolver resolver = context.getContentResolver();
- ContentValues values = new ContentValues();
- long rawContactId = ContentUris.parseId(resolver.insert(RawContacts.CONTENT_URI, values));
- values.clear();
- values.put(Data.RAW_CONTACT_ID, rawContactId);
- values.put(Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE);
- values.put(StructuredName.DISPLAY_NAME, item.getValue(ContactItem.TYPE_NAME));
- resolver.insert(Data.CONTENT_URI, values);
- values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
- values.put(Phone.NUMBER, item.getValue(ContactItem.TYPE_PHONE));
- values.put(Phone.TYPE, Phone.TYPE_MOBILE);
- public void delete(ContactItem item) {
- String[] args = {item.getValue(ContactItem.TYPE_CONTACT_ID)};
- resolver.delete(RawContacts.CONTENT_URI, Data.CONTACT_ID + "=?", args);
- resolver.delete(Data.CONTENT_URI, Data.RAW_CONTACT_ID + "=?", args);
- }
如下是一次性擷取通訊錄清單資訊的類ContractGenerator:
- import java.io.InputStream;
- import java.util.ArrayList;
- import java.util.List;
- import android.database.Cursor;
- import android.graphics.Bitmap;
- import android.graphics.BitmapFactory;
- import android.net.Uri;
- import android.provider.ContactsContract.Contacts;
- import android.provider.ContactsContract.CommonDataKinds.Photo;
- public class ContactGenerator {
- private static final String[] PROJECTECTION = {
- Phone.DISPLAY_NAME,
- Phone.NUMBER,
- Photo.PHOTO_ID,
- Phone.CONTACT_ID,
- };
- private static final int DISPLAY_NAME_INDEX = 0;
- private static final int PHONE_NUMBER_INDEX = 1;
- private static final int PHOTO_ID_INDEX = 2;
- private static final int CONTACT_ID_INDEX = 3;
- public ContactGenerator(Context context) {
- public List<ContactItem> generateList() {
- List<ContactItem> list = new ArrayList<ContactItem>();
- Cursor cursor = resolver.query(Phone.CONTENT_URI, PROJECTECTION, null, null, PROJECTECTION[DISPLAY_NAME_INDEX] + " ASC");
- while (cursor.moveToNext()) {
- String name = cursor.getString(DISPLAY_NAME_INDEX);
- String phone = cursor.getString(PHONE_NUMBER_INDEX);
- long photoId = cursor.getLong(PHOTO_ID_INDEX);
- long contactId = cursor.getLong(CONTACT_ID_INDEX);
- Bitmap head = null;
- if (photoId > 0) {
- Uri uri = ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId);
- InputStream input = Contacts.openContactPhotoInputStream(resolver, uri);
- head = BitmapFactory.decodeStream(input);
- }
- ContactItem item = new ContactItem(context, name, phone, head);
- item.setContactId(contactId);
- list.add(item);
- }
- return list;
對系統通訊錄資料庫進行操作,需要在AndroidManifest.xml檔案中添加如下兩個使用者權限:
[html] view plaincopy
- <uses-permission android:name="android.permission.READ_CONTACTS"></uses-permission>
- <uses-permission android:name="android.permission.WRITE_CONTACTS"></uses-permission>
對于ContentResolver的使用,主要是如下幾個步驟:
1. 通過Context的getContentResolver()方法擷取ContentResolver執行個體;
2. 調用ContentResolver的insert、delete、update、query方法對資料表進行增删改查操作,對資料表的指定是通過系統URI來完成,這些URI會作為參數傳入上述4個方法中。詳情參見Android API文檔。
除了前述兩個對ContentResolver的操作外,本通訊錄的UI控件主要基于二級清單ExpandableListView,是以有繼承自BaseExpandableListAdapter的二級清單資料處理擴充卡類ContactAdapter。另外,ContactItem是資料實體類。其餘兩個Activity類的界面類。
最後需要說明的是,對于系統已經提供了URI的資料資源,沒有必要再用ContentProvider對其進行封裝管理,因而本示例中并沒有用到ContentProvider,隻用到ContentResolver。