Индекс android.database.cursor выходит за пределы исключения?

Я пытаюсь получить имя и тип контактов, хранящихся в телефоне, но получаю это исключение. Я получал имена раньше, но получал исключение

android.database.CursorIndexOutOfBoundsException: запрашивается индекс -1 с размером 10

при попытке получить имя и тип вместе. Пожалуйста помоги. Заранее спасибо.

package application.test;
import android.app.Activity;
import android.content.ContentResolver;
import android.database.Cursor;
import android.os.Bundle;

import android.provider.ContactsContract;
import android.provider.ContactsContract.CommonDataKinds.Phone;
import android.provider.ContactsContract.Contacts;
import android.util.Log;

public class TestActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        String[] projection = new String[] {ContactsContract.Contacts.DISPLAY_NAME};     
        String[] projection1=new String[]{Phone.TYPE};
        String[] projection2=new String[]{ContactsContract.Contacts._ID};

        ContentResolver cr = getContentResolver();
        ContentResolver ncr=getContentResolver();
        ContentResolver icr=getContentResolver();

        Cursor cur = cr.query(ContactsContract.Contacts.CONTENT_URI, projection,null, null, Contacts.DISPLAY_NAME + " ASC");
        Cursor ncur=ncr.query(ContactsContract.Data.CONTENT_URI, projection1, null, null,null);
        Cursor icur = icr.query(ContactsContract.RawContacts.CONTENT_URI, projection2,null, null, Contacts._ID + " ASC");

        if (cur.getCount() >0 && ncur.getCount()>0 && icur.getCount()>0) 
        {
            while (cur.moveToNext()&& ncur.moveToNext()&& icur.getCount()>0) 
            {

                String id = icur.getString(cur.getColumnIndex(ContactsContract.Contacts._ID));
                String name = cur.getString(cur.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME));
                String  type=ncur.getString(ncur.getColumnIndex(Phone.TYPE));

                if (Integer.parseInt(cur.getString(cur.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER))) > 0) 
                {
                    Cursor pCur = cr.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, 
                                                                 ContactsContract.CommonDataKinds.Phone.CONTACT_ID +" = ?",new String[]{id}, null);     
                    Cursor typecur = ncr.query( ContactsContract.CommonDataKinds.Phone.CONTENT_URI,null,
                                                                        ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ?", new String[]{id}, null)          

                    while (pCur.moveToNext()&& typecur.moveToNext())
                    {

                        Log.d("names",name);
                        Log.d("types",type);
                        pCur.close();
                    } 
                }
            }
        }
    }
}

person love_bird    schedule 09.11.2011    source источник
comment
почему вы делаете три отдельных курсора?   -  person Vladimir    schedule 09.11.2011
comment
потому что использую три разные проекции. Я пытаюсь использовать одну, но получаю ошибку времени компиляции.   -  person love_bird    schedule 09.11.2011


Ответы (3)


Когда вы создаете курсор перед его чтением. Переместите курсор на первую позицию.

while (cur.moveToNext()&& ncur.moveToNext()&& icur.getCount()>0) 

  /* move icur to first position then read */
  icur.moveFirst()

Это должно работать

person siva    schedule 09.11.2011
comment
cur.moveToNext() сделает это за вас, она будет действовать как cur.moveToFirst(). - person Carnal; 09.11.2011

Не стоит создавать три разных курсора и contextresolver, используйте один и тот же, но делайте условия в цикле, чтобы получить желаемое, я дам вам метод, который написал сам. Это даст вам все контакты в вашем телефоне, у которых есть номер телефона, и вы поймете, как реализовать свой:

public ArrayList<Contact> getSMSContacts(Context context){
    ArrayList<Contact> contacts = new ArrayList<Contact>();
    ContentResolver cr = context.getContentResolver();
    Cursor cur = cr.query(ContactsContract.Contacts.CONTENT_URI, null, null, null, ContactsContract.Contacts.DISPLAY_NAME + " asc");
    if (cur.getCount() > 0) {
        while (cur.moveToNext()) {
            String id = cur.getString(cur.getColumnIndex(ContactsContract.Contacts._ID));
            String name = cur.getString(cur.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME));
            if (Integer.parseInt(cur.getString(cur.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER))) > 0) {
                Contact c = new Contact();
                c.setId(Integer.parseInt(id));
                c.setName(name);
                Cursor pCur = cr.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
                        null, ContactsContract.CommonDataKinds.Phone.CONTACT_ID +" = ?", 
                        new String[]{id}, null);
                while (pCur.moveToNext()) {
                    String number = pCur.getString(
                            pCur.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
                    c.setNumber(number);
                } 
                pCur.close();
                contacts.add(c);
            }
        }
    }
    cur.close();
    return contacts;
}
person Carnal    schedule 09.11.2011
comment
здесь вы получаете только контакты, которые хранятся в столбце контактов. Для типа это выглядит как Cursor ncur=ncr.query(ContactsContract.Data.CONTENT_URI, project1, null, null,null); поскольку типы оцениваются в столбце данных. так же, как contact_id получает хранилище в столбце rawcontactid. Помимо того, что я получал исключение во время выполнения, используя один объект курсора и передавая ему проекцию. - person love_bird; 09.11.2011
comment
Как я уже говорил, вы поймете, посмотрев мою реализацию, и, кстати, она отлично работает! Что вы имеете в виду под типом? Что вы хотите, кроме имени контакта. - person Carnal; 09.11.2011
comment
Тип строки = pCur.getString(pCur.getColumnIndex(ContactsContract.CommonDataKinds.Phone.TYPE)); - person Carnal; 09.11.2011

После нескольких дней поиска я, наконец, получил ответы. Тип и имя сохраняются в таблице данных в столбце DATA2, поэтому я запросил data.content_uri и получил индекс столбца DATA2, и он просто работает нормально... тип.

person love_bird    schedule 10.11.2011