通讯录数据库分析
在通讯录这个应用里有个特别重要的三个表,分别为:raw_contacts、data、mimetype。
1、raw_contacts表保存联系人的ID 例如:contact_id = 0;
2、data表保存联系人的数据,字段data1存放数据,mimetype_id 存放数据类型,raw_contacts_id字段表示保存数据属于那个人联系人的
3、mimetype表保存数据类型 ,字段mimetype_id = 1 代表email,mimetype_id = 5 代表phone, mimetype_id = 7 代表name
其中通讯录中显示的手机号存放在data表中,而对应的用户显示名称却是在raw_contacts中,所以当我们要去获取手机号对应的用户显示名称的时候,我们还需要去查看raw_contacts表中的值,这个中间对应的桥梁就是data表中raw_contact_id。
raw_contacts是三个表的基础,他的_ID列与data表中的raw_contacts_id关联。
相关uri
在安卓系统中,关于通讯录的URI都已经预先解析完毕,其结果都被当作常量,存在源码里。
例如 “content://com.android.contacts/raw_contacts”,被解析为
ContactsContract.RawContacts.CONTENT_URI
其中ContactsContract.RawContacts指的是raw_contacts表,常量CONTENT_URI为解析出来的结果。
增删改查
查询
1.获取raw_contacts表中的contact_id的所有值,返回的cursor对象中会包含与id相关联的其他两表的信息
2.遍历cursor对象。
注意:在查询输出的时候,使用DISPLAY_NAME列名。在添加或者修改时,传入的是GIven_NAME列
Cursor cursor = resolver.query(ContactsContract.RawContacts.CONTENT_URI, null, null,
null);
while (cursor.moveToNext()) {
//新建一个Bean来存储数据
ContactBean bean = new ContactBean();
int id = cursor.getInt(cursor.getColumnIndex(ContactsContract.RawContacts._ID));
String name = cursor.getString(cursor.getColumnIndex(ContactsContract.Data.DISPLAY_NAME));
//一个用户的手机号可能有多个,故手机号是一个集合。,取第一个手机号
String phone = null;
Cursor phoneCursor = resolver.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
null, ContactsContract.Data.RAW_CONTACT_ID + "=?", new String[]{
String.valueOf(id)}, null);
if (phoneCursor.moveToFirst()) {
phone = phoneCursor.getString(phoneCursor.getColumnIndex(ContactsContract.
CommonDataKinds.Phone.NUMBER));
}
phoneCursor.close();
bean.setId(id);
bean.setName(name);
bean.setPhone(phone);
mList.add(bean);
添加
1.先向raw_contact表中添加一条数据。由于_ID列是自增类型的,新增的数据会自动排到最后一行。
2.向data表中插入对应的值和mimetype类型,raw_contact_id = contact_id;
/**
* 首先向row_contact表中插入一条数据,目的是为了获取raw_contact_id字段
*/
ContentValues values = new ContentValues();
Uri rawContactUri = resolver.insert(ContactsContract.RawContacts.CONTENT_URI,values);
//ContentUris一个uri相关的工具类
long rawContactId = ContentUris.parseId(rawContactUri);//通过uri获取id
/**
* 在row_contact中的一条数据是对应DAta表中的两条数据,故要单独插入。
* 插入姓名
*/
values.clear();
values.put(ContactsContract.Data.RAW_CONTACT_ID,rawContactId);
//MIMETYPE是指文件类型,eg.html...
values.put(ContactsContract.Data.MIMETYPE,ContactsContract.CommonDataKinds.
StructuredName.CONTENT_ITEM_TYPE);
values.put(ContactsContract.CommonDataKinds.StructuredName.GIVEN_NAME,name);
Uri nameUri = resolver.insert(ContactsContract.Data.CONTENT_URI, values);
/**
* 插入手机号
*/
values.clear();
values.put(ContactsContract.Data.RAW_CONTACT_ID,rawContactId);
//MIMETYPE是指文件类型,eg.html...,name与phone对应的表的类型不同
values.put(ContactsContract.Data.MIMETYPE,ContactsContract.CommonDataKinds.
Phone.CONTENT_ITEM_TYPE);
values.put(ContactsContract.CommonDataKinds.Phone.NUMBER,phone);
Uri PhoneUri = resolver.insert(ContactsContract.Data.CONTENT_URI, values);
修改
1.先获取数据的id
2.以id为筛选条件,进行修改
ContentValues values = new ContentValues();
values.put(ContactsContract.CommonDataKinds.StructuredName.GIVEN_NAME, name);
values.put(ContactsContract.CommonDataKinds.Phone.NUMBER,phone);
Log.d("updateContact", "values已经产生");
int result = resolver.update(ContactsContract.Data.CONTENT_URI, values,
ContactsContract.Data.RAW_CONTACT_ID +"=?",new String[]{
String.valueOf(bean.getId())});
删除
1.获取id
2.以id作为筛选条件进行删除。
由于raw_contacts是三个表的基础,他的_ID列与data表中的raw_contacts_id关联。所有只要删了raw_contacts的id,与之相关的数据会自动删除。
int result = resolver.delete(ContactsContract.RawContacts.CONTENT_URI,
ContactsContract.RawContacts._ID + "=?",
new String[]{String.valueOf(bean.getId())});