NullPointerException в onLoaderFinished с использованием SimpleCursorAdapter

Я переключился с ResourceCursorAdapter, где использовал newView и bindView в SimpleCursorAdapter, где я использую только метод getView.

Теперь у меня есть ошибка в onLoaderFinished. Хотя он дает мне NullPointerException на adapter.swapCursor(cursor), оба объекта адаптера и курсора NOT null. Я напишу весь свой код ниже. Любая помощь очень ценится (не хватает много волос, чтобы вытащить).

import android.annotation.SuppressLint;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.LoaderManager;
import android.support.v4.content.CursorLoader;
import android.support.v4.content.Loader;
import android.support.v4.widget.ResourceCursorAdapter;
import android.util.Log;
import android.util.SparseBooleanArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.view.Window;
import android.widget.CheckBox;
import android.widget.ListView;
import android.widget.TextView;

public class ContactSelect extends FragmentActivity implements LoaderManager.LoaderCallbacks<Cursor> {
private static final int LOADER_ID = 1;
private MyAdapter adapter;
private ListView list;
private View row;
private SparseBooleanArray checkedState = new SparseBooleanArray();

@SuppressLint({ "NewApi", "NewApi" })
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    requestWindowFeature(Window.FEATURE_INDETERMINATE_PROGRESS);
    setContentView(R.layout.activity_contact_select);       

    adapter = new MyAdapter(this, R.layout.contacts_select_row, null, null, null, 0);     

    getSupportLoaderManager().initLoader(LOADER_ID, null, this);        

    list = (ListView)findViewById(R.id.list);                 

    list.setAdapter(adapter);
    list.setEmptyView(findViewById(R.id.empty));    

}   

@SuppressLint("NewApi")
public Loader<Cursor> onCreateLoader(int arg0, Bundle arg1) {
    final String projection[] = new String[]{ContactsContract.Contacts._ID, ContactsContract.Contacts.DISPLAY_NAME};
    final Uri uri = ContactsContract.Contacts.CONTENT_URI;

    final String selection = ContactsContract.Contacts.HAS_PHONE_NUMBER + "=1" + 
    " AND " + ContactsContract.Contacts.IN_VISIBLE_GROUP + " =1";

    final String order = ContactsContract.Contacts.DISPLAY_NAME + " COLLATE LOCALIZED ASC";

    final CursorLoader loader = new CursorLoader(this, uri, projection, selection, null, order);

    return loader;      
}

public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
    for(int i=0;i<cursor.getCount();i++){
        checkedState.put(i, false);
    }

    adapter.swapCursor(cursor);                 
}

public void onLoaderReset(Loader<Cursor> loader) {
    adapter.swapCursor(null);       
}

private class MyAdapter extends SimpleCursorAdapter implements OnClickListener{
    private CheckBox markedBox;
    private TextView familyText;
    private Context context;
    private Cursor cursor;

    public MyAdapter(Context context, int layout, Cursor c, String[] from,
            int[] to, int flags) {
        super(context, layout, c, from, to, flags);

        this.context = context;
        this.cursor = getCursor();
    }

    @Override
    public View getView(int position, View view, ViewGroup group) {

        final LayoutInflater li = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        row = li.inflate(R.layout.contacts_select_row, group, false);

        view.setTag(cursor.getPosition());
        view.setOnClickListener(this);

        familyText = (TextView)view.findViewById(R.id.contacts_row_family_name);
        markedBox = (CheckBox)view.findViewById(R.id.contacts_row_check);
        familyText.setText(cursor.getString(cursor.getColumnIndexOrThrow(ContactsContract.Contacts.DISPLAY_NAME)));

        boolean currentlyChecked = checkedState.get(cursor.getPosition());
        markedBox.setChecked(currentlyChecked);     


        setProgressBarIndeterminateVisibility(false);

        return super.getView(position, view, group);
    }

    public void onClick(View view) {
        int rowId = (Integer)view.getTag();
        Log.d("OnClick", String.valueOf(rowId));
        boolean currentlyChecked = checkedState.get(rowId);
        markedBox.setChecked(!currentlyChecked);
        checkedState.put(rowId, !currentlyChecked);
        Log.d("checkedState", "checkedState(" + rowId + ") = " + checkedState.get(rowId));
    }       
      }     
}

Ответ 1

Вызов метода swapCursor класса SimpleCursorAdapter вызовет функцию, которая отображает имена столбцов из массива String, предоставленные конструктору (4-й параметр) в массив целых чисел, представляющих индексы столбцов, Когда вы передаете null в конструкторе MyAdapter для массива String, представляющего имена столбцов из курсора, это приведет к тому, что NullPointerException будет позже, когда swapCursor попытается сделать сопоставление ( должен отображаться в методе findColumns, который является фактическим методом, который использует массив имен столбцов String).

Решение состоит в том, чтобы передать действительный массив String, вы также можете сделать это для массива int, представляющего идентификаторы для представлений, в которые помещать данные:

String[] from = {ContactsContract.Contacts.DISPLAY_NAME};
int[] to = {R.id.contacts_row_family_name, R.id.contacts_row_check};
adapter = new MyAdapter(this, R.layout.contacts_select_row, null, from, to, 0);

Я не знаю, что вы пытаетесь сделать, но ваша реализация метода getView не совсем правильная:

Вы делаете обычный материал для метода getView (создаете макеты, просматриваете представления, привязываете данные), а затем просто возвращаете представление из суперкласса (?!?), вы, вероятно, просто увидите макет по умолчанию с ничего в нем.

Способ, которым вы написали метод getView, не очень эффективен, вы можете захотеть взглянуть на переработку вида и шаблон держателя вида.

cursor.getPosition() не будет делать то, что вы хотите, поскольку вы не переместите курсор в правильное положение. По умолчанию адаптеры, основанные на курсорах, сделают это для вас в методе getView, но, поскольку вы переопределили метод, ваше задание перемещает позицию курсора.

Вы должны оставить метод getView и использовать два метода newView и bindView, поскольку они предлагают лучшее разделение логики.

Ответ 2

adapter = new MyAdapter (это, R.layout.contacts_select_row, null, null, null, 0);

и ваш класс MyAdapter вы пропускаете нулевой курсор

  public MyAdapter(Context context, int layout, Cursor c, String[] from,
            int[] to, int flags) {
        super(context, layout, c, from, to, flags);

        this.context = context;
        this.cursor = getCursor();
    }