Получение нулевого указателя на свопкурсоре после длительной неактивности
Мои загрузчики, фрагменты и контент-провайдер работают нормально. Все мои списки могут обновляться в порядке, но после длительного периода бездействия (основная активность приостанавливается), если я повторно активирую приложение, я получаю нулевой указатель на вызов 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
, который вызывает исключение нулевого указателя позже, когда к нему обращаются.