CursorLoaderを使用してコンテンツを取得するListFragmentsがいくつかあります。ユーザーがコンテンツを掘り下げていくうちに、あるFragmentが別のFragmentを置き換えます(Activityは同じままです)。しかし、非最上位の断片上のコンテンツの変更、アプリがクラッシュした場合:バックグラウンド更新は内容を変更したとき、またはユーザーが戻るホーム画面から、別のフラグメントが一番上にあるときにAndroid:最上部以外の部分でCursorLoaderがクラッシュする
E/AndroidRuntime(18830): FATAL EXCEPTION: main
E/AndroidRuntime(18830): java.lang.RuntimeException: Unable to resume activity {com.example.ExampleApp/com.example.ExampleApp.ExampleActivity}: java.lang.IllegalStateException: Content view not yet created
E/AndroidRuntime(18830): at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2120)
E/AndroidRuntime(18830): at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:2135)
E/AndroidRuntime(18830): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:957)
E/AndroidRuntime(18830): at android.os.Handler.dispatchMessage(Handler.java:99)
E/AndroidRuntime(18830): at android.os.Looper.loop(Looper.java:130)
E/AndroidRuntime(18830): at android.app.ActivityThread.main(ActivityThread.java:3683)
E/AndroidRuntime(18830): at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime(18830): at java.lang.reflect.Method.invoke(Method.java:507)
E/AndroidRuntime(18830): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
E/AndroidRuntime(18830): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
E/AndroidRuntime(18830): at dalvik.system.NativeStart.main(Native Method)
E/AndroidRuntime(18830): Caused by: java.lang.IllegalStateException: Content view not yet created
E/AndroidRuntime(18830): at android.support.v4.app.ListFragment.ensureList(ListFragment.java:328)
E/AndroidRuntime(18830): at android.support.v4.app.ListFragment.setListShown(ListFragment.java:280)
E/AndroidRuntime(18830): at android.support.v4.app.ListFragment.setListShownNoAnimation(ListFragment.java:266)
E/AndroidRuntime(18830): at com.example.ExampleApp.FirstListFragment.onLoadFinished(FirstListFragment.java:102)
E/AndroidRuntime(18830): at com.example.ExampleApp.FristListFragment.onLoadFinished(FirstListFragment.java:20)
E/AndroidRuntime(18830): at android.support.v4.app.LoaderManagerImpl$LoaderInfo.callOnLoadFinished(LoaderManager.java:414)
E/AndroidRuntime(18830): at android.support.v4.app.LoaderManagerImpl$LoaderInfo.reportStart(LoaderManager.java:298)
E/AndroidRuntime(18830): at android.support.v4.app.LoaderManagerImpl.doReportStart(LoaderManager.java:751)
E/AndroidRuntime(18830): at android.support.v4.app.FragmentActivity.onStart(FragmentActivity.java:512)
E/AndroidRuntime(18830): at android.app.Instrumentation.callActivityOnStart(Instrumentation.java:1129)
E/AndroidRuntime(18830): at android.app.Activity.performStart(Activity.java:3791)
E/AndroidRuntime(18830): at android.app.Activity.performRestart(Activity.java:3821)
E/AndroidRuntime(18830): at android.app.Activity.performResume(Activity.java:3826)
E/AndroidRuntime(18830): at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2110)
E/AndroidRuntime(18830): ... 10 more
これが発生する可能性がありますバックスタックのonLoadFinished
ハンドラは、存在しなくなったUIを更新しようとします。しかし、なぜフラグメントはまだカーソルの更新を得ているのですか?
フラグメントが表示されていない場合は更新を中止することで回避しましたが、これは間違っているようです。簡潔にするために少し編集ここ
は、私の断片は次のようになります。
public class FirstListFragment extends ListFragment implements LoaderManager.LoaderCallbacks<Cursor> {
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
// Initialize empty cursor adapter.
mAdapter = new ExampleCursorAdapter(getActivity(), null, 0);
setListAdapter(mAdapter);
// Start with a progress indicator
setListShown(false);
// Prepare the loader. Either re-connect with an existing one, or start a new one.
getLoaderManager().initLoader(EXAMPLE_LOADER_ID, null, this);
}
@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
return new CursorLoader(getActivity(), ExampleProvider.CONTENT_URI, PROJECTION, null, null, null);
}
@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
// Do nothing if we're not visible.
if(!isVisible()) {
return;
}
// Swap the new cursor in. (The framework will take care of closing the
// old cursor once we return.)
mAdapter.swapCursor(data);
// The list should now be shown.
if(isResumed()) {
setListShown(true);
} else {
setListShownNoAnimation(true);
}
}
@Override
public void onLoaderReset(Loader<Cursor> loader) {
// This is called when the last Cursor provided to onLoadFinished() above is about to be
// closed. We need to make sure we are no longer using it.
mAdapter.swapCursor(null);
}
}
isVisible()
チェックはクラッシュを防ぎ、それは私が見てきた例のコードのいずれかではありません。ここで何が間違っていますか?
編集:十分確かに、私は、閉じたカーソルを使用しようとしている古いビューでStaleDataExceptionで巻き取ります。今はビューが破棄されたときにLoaderManagerを破棄しています。これが正しいことであるかどうかまだ分かりませんし、StaleDataExceptionを再現することはできません。
私は上からisVisible()
ハックを削除し、これを追加しました:
@Override
public void onDestroyView() {
super.onDestroyView();
// The CursorLoader example doesn't do this, but if we get an update while the UI is
// destroyed, it will crash. Why is this necessary?
getLoaderManager().destroyLoader(EXAMPLE_LOADER_ID);
}
ええ、ローダーがあまりにも早く戻り、getActivity()がまだヌル(完全には添付されていない)のバグがあるようです。ローダーは時にはフレームワークによって再起動されるようです。あまりにも早く戻る原因になります。 – pjco