2011-07-20 12 views
14

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); 
} 

答えて

2

これは、リストのフラグメントがリストビューを取得する際に問題があると思われます。何らかの理由で、フラグメントが表示されていない限り、getListView()は有効なビューを返しません。私はそれがonStart()の前にライフサイクルの何かに違法な状態の例外をスローするのを見ました。

私が問題を解決する方法は、onCreateView()メソッドでローダーを呼び出すことです。ローダーがデータを返すと、アクティビティーが開始されているかどうかがチェックされます。それがリストビューにロードされていない場合は、それをローカル変数に格納し、onStart()メソッドでデータをロードします。

これはスーパーハックのようですが、機能します。

+0

ええ、ローダーがあまりにも早く戻り、getActivity()がまだヌル(完全には添付されていない)のバグがあるようです。ローダーは時にはフレームワークによって再起動されるようです。あまりにも早く戻る原因になります。 – pjco

関連する問題