0

向きを変更すると IllegalStateException が発生します。SimpleCursorAdapter とContentProvider を使用したlistViewがあります。この例外の原因は何ですか?

編集

アクションバーのスピナーからアイテムを選択して向きを変更するときにのみ例外が表示されるため、何かを変更したに違いありません。アクション バー スピナーには、[すべて表示]、[日付を表示]、[場所を表示] の 3 つの項目があります。

ユーザーがいずれかを選択すると、データベースにクエリが実行されます (onNavigationItemSelected() を参照)。onStop() でカーソルを閉じようとしましたが、問題は解決しませんでした。それを閉じるのに適切な場所はどこでしょうか?

MainActivity で:

 private  Cursor mCursor = null;

 public void onCreate() {
    mSimpleCursorAdapter = new SpecialAdapter(this, 
            R.layout.row,
            null,
            //cursor,
            PROJECTION,
            new int[] { R.id.titleID, R.id.dateTimeOrLocationID1 , R.id.dateTimeOrLocationID2 },
            CursorAdapter.NO_SELECTION);

    mListView = (ListView) findViewById(android.R.id.list);
    mListView.setAdapter(mSimpleCursorAdapter);


    mOnNavigationListener = new OnNavigationListener() {

          @Override
          public boolean onNavigationItemSelected(int position, long itemId) {

              switch(position) { 
              case 0:
                  mCursor = getContentResolver().query(ReminderContentProvider.CONTENT_URI, PROJECTION, null, null, null);
                  break;
              case 1:

                  mCursor = getContentResolver().query(ReminderContentProvider.CONTENT_URI, PROJECTION, " Date NOT NULL", null, null);
                  break;
              case 2:
                  mCursor = getContentResolver().query(ReminderContentProvider.CONTENT_URI, PROJECTION, " Address NOT NULL", null, null);
                  break;
              default:
                  break;
              }
            getLoaderManager().restartLoader(0, null, MainActivity.this);
            return true;
          }
        };
}

@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
    CursorLoader loader = new CursorLoader(this, ReminderContentProvider.CONTENT_URI, PROJECTION, null, null, null);
    return loader;
}

@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
    mSimpleCursorAdapter.swapCursor(data);
}

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

SimpleCursorAdapter:

@Override
public View getView(int position, View convertView, ViewGroup parent) {

    Cursor mCursor = (Cursor) getItem(position); // exception
    if(mCursor != null)
    {
              ........
    }
 }

編集

E/ACRA    ( 3348): com.example.locationreminder fatal error : attempt to re-open an already-closed   object: SQLiteQuery: SELECT _id, Title, Date, Address, Radius, Repetition FROM reminder
E/ACRA    ( 3348): java.lang.IllegalStateException: attempt to re-open an already-closed object: SQLiteQuery: SELECT _id, Title, Date, Address, Radius, Repetition FROM reminder
E/ACRA    ( 3348):  at android.database.sqlite.SQLiteClosable.acquireReference(SQLiteClosable.java:55)
E/ACRA    ( 3348):  at android.database.sqlite.SQLiteQuery.fillWindow(SQLiteQuery.java:58)
E/ACRA    ( 3348):  at android.database.sqlite.SQLiteCursor.fillWindow(SQLiteCursor.java:151)
E/ACRA    ( 3348):  at android.database.sqlite.SQLiteCursor.onMove(SQLiteCursor.java:124)
E/ACRA    ( 3348):  at android.database.AbstractCursor.moveToPosition(AbstractCursor.java:213)
E/ACRA    ( 3348):  at android.database.CursorWrapper.moveToPosition(CursorWrapper.java:162)
E/ACRA    ( 3348):  at android.widget.CursorAdapter.getItem(CursorAdapter.java:207)
E/ACRA    ( 3348):  at com.example.locationreminder.SpecialAdapter.getView(SpecialAdapter.java:51)
E/ACRA    ( 3348):  at android.widget.AbsListView.obtainView(AbsListView.java:2271)
E/ACRA    ( 3348):  at android.widget.ListView.makeAndAddView(ListView.java:1769)
4

1 に答える 1

1

これが失敗の理由かどうかはわかりませんが、コードに明らかにバグがあります。手動のカーソル管理とローダーを混在させようとしています。OnNavigationItemSelectedあなたの呼び出しでchangeCursor。いくつかの問題があります:

  1. UIスレッドでデータをロードします。これを回避するためのローダーが存在します
  2. changeCursor古いカーソルを閉じます。ただし、このカーソルはローダーによって所有されているため、閉じないでください。呼び出した後、ローダーによって閉じられますOnLoadFinished
  3. 手動で作成されたカーソルがどこで閉じられるかは明確ではありません。

ですべきことはOnNavigationItemSelected、新しいクエリ パラメータでローダーを再起動することです。

于 2013-08-21T02:31:02.097 に答える