联系人数据库读取

android读取联系人是通过contentprovider的接口形式来进行的,而且联系人操作属于高危级别的行为,因此需要做权限申请:

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


  如果是Android6.0以上系统,还需要在实际操作前申请运行时权限,关于运行时权限的问题这里暂且不讨论。
  那么我们首先获取获取一个ContentResolver对象,一般都是Context.getContentResolver()方法,然后执行ContentResolver的query()方法即可获取一个Cursor对象,这个对象里面存储了我们从对应数据库表查出来的所有数据。
  我们首先分析一下query方法中的几个参数,
  第一个参数Uri,这个参数是指定查询的数据位置,一般都是指定到某个表,
  第二个参数String[],这个参数表示我们需要接口访问表中的哪些字段的数据,
  第三个参数String,这个是查询条件,相当于where语句,
  第四个参数String[],这个是查询值。比如你的第三个参数写的是”=?”,那么第四个参数就是表示符合条件的情况的集合,
  第五个参数String,表示排序条件。
  如果我们需要获取一张表的所有数据,那么我们可以将query方法中除Uri以外的所有参数都设为null。那么接下来,我们就首先来获取contacts表中的数据,这个表相当于联系人数据库中的user表,在android的数据库中,联系人的一些详细信息并不存在于contacts表中,而且存在其它表里面,contacts就是存储其他表中的一些id来关联这些数据。查询的方法如下:  

Cursor cursor = getContentResolver().query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null);


  以上这个cursor对象里面就存储了contacts表中的所有信息,接下来就是根据contacts保存的关联信息来获取联系人的一些实际信息了。
  我们首先来获取联系人的姓名相关的信息,姓名相关的信息主要是存储在raw_contacts表中,该表的URI是ContactsContract.RawContacts.CONTENT_URI。
  然后是关键的电话号码信息,电话号码的信息存储在data表中,这个表其实不只存储了电话号码,还包括姓名、邮箱、社交帐号、家庭住址等等信息。而且这里面的数据不是一个联系人对应的一条数据,而是多个联系人对应有多条数据。我们可以根据该表中的mimetype_id 字段来区分这条数据究竟是姓名信息还是电话信息或者邮箱信息。具体的数据则保存在data1 到data15 这15个字段中。
  以上分析就到此为止,下面是具体的代码:

ContentResolver resolver = getContentResolver();
        Cursor cursor = resolver.query(ContactsContract.RawContacts.CONTENT_URI, null, null, null, null);
        if (cursor != null && cursor.getCount() > 0) {
            cursor.moveToFirst();
            while (cursor.moveToNext()) {
                int userId = cursor.getInt(cursor.getColumnIndex(ContactsContract.Contacts._ID));
                Cursor dataCursor = resolver.query(ContactsContract.Data.CONTENT_URI, null, ContactsContract.Data.RAW_CONTACT_ID + "=" + userId, null, null);
                if (dataCursor != null && dataCursor.getCount() > 0) {
                    dataCursor.moveToFirst();
                    while (dataCursor.moveToNext()) {
                        String mimeType = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.MIMETYPE));
                        //电话号码
                        if ("vnd.android.cursor.item/phone_v2".equals(mimeType)) {
                            //这里DATA1和DATA4都存储的是电话号码 但是DATA1的数据里面有短横线或者空格
                            //注意 这里phone可能存在多个 后面的内容也有可能为空
                            String phone = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DATA4));
                        //姓名
                        } else if ("vnd.android.cursor.item/name".equals(mimeType)) {
                            //DATA1表示姓名全称
                            String displayName = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DATA1));
                            //DATA2表示名
                            String firstName = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DATA2));
                            //DATA3表示姓
                            String lastName = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DATA3));
                        //电子邮箱
                        } else if ("vnd.android.cursor.item/email_v2".equals(mimeType)) {
                            String email = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DATA1));
                        }
                        //这里mimeType有很多种类型 这里就暂时例举了三种 如果想了解更多类型
                        // 可以进入系统/data/data/com.android.providers.contacts/databases/目录下面
                        //提取一个名为contacts2.db的数据库文件 查看mimetypes表和data表
                    }
                }
                if (dataCursor!=null){
                    dataCursor.close();
                }
            }
        }
        if (cursor!=null){
            cursor.close();
        }

以上是一种查询方式,但是这种查询方式会有点麻烦,下面推荐一种更加简单的查询方式:

 ContentResolver resolver = getContentResolver();
        Cursor cursor = resolver.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null, null, ContactsContract.CommonDataKinds.Phone.RAW_CONTACT_ID);
        if (cursor != null) {
            cursor.moveToFirst();
            while (cursor.moveToNext()) {
                //注意 如果某个联系人有多个电话 这里会出现多条 rawId相同 但是phone不同的数据
                int rawId = cursor.getInt(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.RAW_CONTACT_ID));
                String phone = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NORMALIZED_NUMBER));
                String name = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
            }
            cursor.close();
        }


————————————————
版权声明:本文为CSDN博主「余烬岛游戏」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/w815878564/article/details/52474615

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值