Как получить все данные контактов от контент-провайдера с помощью одного запроса?

Мне нужно прочитать контакты с устройства. Требуемые поля: ID , Display Name, Phone Number (все типы) Email id (все типы) и «фото». Для этого прямо сейчас я делаю так.

1: Сначала я читаю все id из ContactsContract.Contacts.CONTENT_URI;, как показано ниже.

        Uri contactsUri = ContactsContract.Contacts.CONTENT_URI;
        // Querying the table ContactsContract.Contacts to retrieve all the contacts
        String[] projection = {ID};
        Cursor contactsCursor = mContentResolver.query(contactsUri, projection, null, null,
                "upper(" + ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME + ") ASC");

2 : Затем перебираем этот курсор, чтобы прочитать обязательные поля всех контактов.

if (contactsCursor.moveToFirst()) {
            do {
                 long contactId = contactsCursor.getLong(contactsCursor.getColumnIndex(ID));
                   Uri dataUri = ContactsContract.Data.CONTENT_URI;
                // Querying the table ContactsContract.Data to retrieve individual items like
                // home phone, mobile phone, work email etc corresponding to each contact
                   String[] columns = {CONTACT_ID, PHOTO_URI, DISPLAY_NAME, MIME_TYPE, DATA_1, DATA_2, DATA_4};
                   String selection = ContactsContract.Data.CONTACT_ID + "=" + contactId;
                   Cursor dataCursor = mContentResolver.query(dataUri, columns,
                        selection,
                        null, null);
                                 if (dataCursor.moveToFirst()) {
                                 // Getting Display Name
                                 displayName = dataCursor.getString(dataCursor.getColumnIndex(ContactsContract.Data.DISPLAY_NAME));
                                 do {
                                    // Getting Phone numbers
                                    String mimeType = dataCursor.getString(dataCursor.getColumnIndex(MIME_TYPE));
                                    switch (mimeType) {
                                             case ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE:
                                             String phoneNumber = dataCursor.getString(dataCursor.getColumnIndex(DATA_1));
                                             switch (dataCursor.getInt(dataCursor.getColumnIndex(DATA_2))) {
                                    case ContactsContract.CommonDataKinds.Phone.TYPE_HOME:
                                        homePhone = phoneNumber;
                                        break;
                                    case ContactsContract.CommonDataKinds.Phone.TYPE_MOBILE:
                                        mobilePhone = phoneNumber;
                                        break;
                                    case ContactsContract.CommonDataKinds.Phone.TYPE_WORK:
                                        workPhone = phoneNumber;
                                        break;
                                    default:
                                        otherPhone = phoneNumber;
                                }
                                break;
                              // Getting emails
                              case ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE:
                                switch (dataCursor.getInt(dataCursor.getColumnIndex(DATA_2))) {
                                    case ContactsContract.CommonDataKinds.Email.TYPE_HOME:
                                        homeEmail = dataCursor.getString(dataCursor.getColumnIndex(DATA_1));
                                        break;
                                    case ContactsContract.CommonDataKinds.Email.TYPE_WORK:
                                        workEmail = dataCursor.getString(dataCursor.getColumnIndex(DATA_1));
                                        break;
                                    default:
                                        otherEmail = dataCursor.getString(dataCursor.getColumnIndex(DATA_1));
                                }
                                break;
                               // getting photo Uri 
                               case ContactsContract.CommonDataKinds.Photo.CONTENT_ITEM_TYPE:
                                if (dataCursor.getString(dataCursor.getColumnIndex(PHOTO_URI)) != null) {
                                    photoUri = Uri.parse(dataCursor.getString(dataCursor.getColumnIndex(PHOTO_URI)));
                                }
                                break;
                        }

                    } while (dataCursor.moveToNext());


               } while (contactsCursor.moveToNext());

Запросы работают нормально, но проблема в том, что для повторения и получения подробной информации о каждом контакте требуется слишком много времени. Первый запрос возвращается быстро, но вся задержка теперь во второй части, то есть в итерации по каждой строке первого запроса и запросе для каждого поля. Можно ли это сделать в одном запросе на соединение, чтобы оптимизировать производительность?


person Sarath Kn    schedule 12.03.2016    source источник
comment
ContentProvider и ContentResolver не поддерживают JOIN, отчасти потому, что не требуется, чтобы провайдер поддерживался базой данных SQL или чем-то еще, что понимает, что такое объединение.   -  person CommonsWare    schedule 12.03.2016


Ответы (1)


Да, вы можете сделать это в одном запросе — все данные о контактах фактически хранятся в одной таблице Data, которая содержит все, что вам нужно для контакта, включая CONTACT_ID и DISPLAY_NAME, которые также размещены в таблице Contacts.

Таким образом, вам нужно в основном запросить все из таблицы данных в одном запросе и отсортировать его по контактам, используя HashMap из CONTACT_ID в какой-либо контактный объект (или, если вы предпочитаете список деталей).

(Примечание: убедитесь, что вы импортируете следующие классы только из пакета ContactsContract)

Map<Long, List<String>> contacts = new HashMap<Long, List<String>>();

String[] projection = {Data.CONTACT_ID, Data.DISPLAY_NAME, Data.MIMETYPE, Data.DATA1, Data.DATA2, Data.DATA3, Data.PHOTO_ID};
String selection = Data.MIMETYPE + " IN ('" + Phone.CONTENT_ITEM_TYPE + "', '" + Email.CONTENT_ITEM_TYPE + "')";
Cursor cur = cr.query(Data.CONTENT_URI, projection, selection, null, null);

while (cur != null && cur.moveToNext()) {
    long id = cur.getLong(0);
    String name = cur.getString(1);
    String mime = cur.getString(2); // email / phone / company
    String data = cur.getString(3); // the actual info, e.g. +1-212-555-1234
    int type = cur.getInt(4); // a numeric value representing type: e.g. home / office / personal
    String label = cur.getString(5); // a custom label in case type is "TYPE_CUSTOM"
    long photoId = cur.getLong(6);

    String kind = "unknown";
    String labelStr = "";

    switch (mime) {
        case Phone.CONTENT_ITEM_TYPE: 
            kind = "phone"; 
            labelStr = Phone.getTypeLabel(getResources(), type, label);
            break;
        case Email.CONTENT_ITEM_TYPE: 
            kind = "email";
            labelStr = Email.getTypeLabel(getResources(), type, label);
            break;
    }
    Log.d(TAG, "got " + id + ", " + name + ", " + kind + " - " + data + " (" + labelStr + ")");

    // add info to existing list if this contact-id was already found, or create a new list in case it's new
    List<String> infos;
    if (contacts.containsKey(id)) {
        infos = contacts.get(id);
    } else {
        infos = new ArrayList<String>();
        infos.add("name = " + name);
        infos.add("photo = " + photoId);
        contacts.put(id, infos);
    }
    infos.add(kind + " = " + data + " (" + labelStr + ")");
}
person marmor    schedule 06.06.2017