Получение нулевого указателя на свопкурсоре после длительной неактивности

Мои загрузчики, фрагменты и контент-провайдер работают нормально. Все мои списки могут обновляться в порядке, но после длительного периода бездействия (основная активность приостанавливается), если я повторно активирую приложение, я получаю нулевой указатель на вызов swapsursor

public class ListeFragment extends ListFragment implements LoaderManager.LoaderCallbacks<Cursor> {
    private ListItemSelectListener listeSelectListener;

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            listeSelectListener = (ListItemSelectListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString()
            + " doit implementer ListItemSelectListener");
        }
    }

    // public void onSaveInstanceState (Bundle outState) {
    // super.onSaveInstanceState(outState);
    // int scroll = this.getSelectedItemPosition();
    // outState.putInt("POS", scroll); }

    @Override
    public void onListItemClick(ListView l, View v, int position, long id) {
        super.onListItemClick(l, v, position, id);
        Log.e ("Resto", "pos="+position+" Rowid= "+id);
        listeSelectListener.onItemSelected(position, id);      
    }

    private static final int RESTO_RECENT_LOADER = 0x01;
    private static final int RESTO_ALPHA_LOADER = 0x02;
    private static final int RESTO_HIGH_LOADER = 0x03;
    private static final int RESTO_SEARCH_LOADER = 0x04;
    // private static final String AUTHORITY = "ca.usimage.resto.RestoProvider";
    // private static final String RESTOS_BASE_PATH = "restos";

    private SimpleCursorAdapter adapter;

    public void afficheList(int loader_id, String query) {
        Log.e ("Resto", "loader_id= "+loader_id);
        Bundle mBundle = new Bundle();
        mBundle.putString("search_query", query);
        getLoaderManager().restartLoader(loader_id, mBundle, this);
    }

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        setRetainInstance(true);
        if (savedInstanceState == null) {
            String[] uiBindFrom = { RestoDatabase.COL_ETAB, RestoDatabase.COL_MONTANT };
            int[] uiBindTo = { R.id.TextView01, R.id.Montant };
            // default loader on startup is RECENT_LOADER
            LoaderManager lm = getLoaderManager();
            if (lm.getLoader(RESTO_RECENT_LOADER) != null) {
                lm.initLoader(RESTO_RECENT_LOADER, null, this);
            }

            // if (adapter == null) {
            adapter = new SimpleCursorAdapter(
                getActivity().getApplicationContext(), R.layout.row,
                null, uiBindFrom, uiBindTo,
                CursorAdapter.FLAG_REGISTER_CONTENT_OBSERVER);
            setListAdapter(adapter);
            // }
            // this.setSelection(savedInstanceState.getInt("POS"));
        }
    }

    public Loader<Cursor> onCreateLoader(int id, Bundle args) {
        String[] projection = { RestoDatabase.ID, RestoDatabase.COL_ETAB,  RestoDatabase.COL_MONTANT };
        switch (id){
        case RESTO_RECENT_LOADER:
            return new CursorLoader(getActivity(),
                RestoProvider.CONTENT_URI, projection, null, null, "date_infraction DESC");

        case RESTO_ALPHA_LOADER:
            return new CursorLoader(getActivity(),
                RestoProvider.CONTENT_URI_GROUPBY, projection, null, null,"etablissement ASC");

        case RESTO_HIGH_LOADER:
            return new CursorLoader(getActivity(),
                RestoProvider.CONTENT_URI, projection, null, null, "montant DESC");

        case RESTO_SEARCH_LOADER:
            return new CursorLoader(getActivity(),
                RestoProvider.CONTENT_URI, projection, "etablissement like \"%" + args.getString("search_query") + "%\"", null, "etablissement ASC");

        default: return null;
        }
    }

    public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
        adapter.swapCursor(cursor);
        // position cursor at top of list
        // this.setSelection(0);
    }

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

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

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
  ;
        setContentView(R.layout.liste);
        final ActionBar ab = getActionBar();

        ab.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
        ab.setDisplayOptions(0, ActionBar.DISPLAY_SHOW_TITLE);

        if (savedInstanceState != null){
            tab_pos = savedInstanceState.getInt("tabState");               
            ab.addTab(ab.newTab().setText(R.string.tab_recente).setTabListener(this),0,false);
            ab.addTab(ab.newTab().setText(R.string.tab_alpha).setTabListener(this),1,false);
            ab.addTab(ab.newTab().setText(R.string.tab_fortes).setTabListener(this),2,false);
            ab.setSelectedNavigationItem(tab_pos);       
        } else {       
            ab.addTab(ab.newTab().setText(R.string.tab_recente).setTabListener(this),0,true);
            ab.addTab(ab.newTab().setText(R.string.tab_alpha).setTabListener(this),1,false);
            ab.addTab(ab.newTab().setText(R.string.tab_fortes).setTabListener(this),2,false);
        }

    public void onTabReselected(Tab tab, FragmentTransaction ft) {
        // position cursor at top of list if user retaps a tab
        ListeFragment listeFrg =  (ListeFragment)getFragmentManager().findFragmentById(R.id.listeFragment);
        listeFrg.setSelection(0);
    }

    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        // TODO Auto-generated method stub

        ListeFragment listeFrg = (ListeFragment)getFragmentManager().findFragmentById(R.id.listeFragment);

        int position = tab.getPosition();
        Log.e ("resto", "pos= "+position);
        switch (position) {
        case 0:        
            listeFrg.afficheList(RESTO_RECENT_LOADER, null);
            break;
        case 1:
            listeFrg.afficheList(RESTO_ALPHA_LOADER, null);
            break;
        case 2:
            listeFrg.afficheList(RESTO_HIGH_LOADER, null);
            break;
        }           
    }

    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        // TODO Auto-generated method stub    
    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putInt("tabState", getActionBar().getSelectedTab().getPosition());
    }

1 ответ

Решение

Ваша проблема, скорее всего, в onActivityCreated,

После длительного периода бездействия операционная система разрушает вашу деятельность. Когда вы пытаетесь вернуться к своей деятельности, она воссоздается. После onCreated деятельности называется, onActivityCreated фрагмента называется, но у вас есть условие, которое создает adapter только если savedInstanceState нулевой. Если это не нуль, вы не создаете adapter, который вызывает исключение нулевого указателя позже, когда к нему обращаются.

Другие вопросы по тегам