【问题标题】:Cursor finalized without prior close() in ListFragment光标在 ListFragment 中没有事先 close() 的情况下完成
【发布时间】:2013-07-07 14:27:28
【问题描述】:

我在关闭应用程序或让手机在应用程序打开的情况下进入睡眠状态后收到此错误消息。

当我让手机进入睡眠状态时,我会收到 30 到 61 个有关此错误消息的日志条目。在加载 ListFragment 之前记录错误。我怀疑片段已加载,但在片段的信息条目之前记录了错误。

当我关闭应用程序时,我只收到 1 个错误条目,该条目是在 onLoaderReset() 调用之后记录的。我在应用程序关闭之前关闭了数据库,这似乎有帮助。

该应用程序可以运行,但我担心它一旦在各种手机上运行就会失败。

除了onLoaderReset()中的adapter.swapCursor(null),我没有显式控制光标。

感谢任何关于如何解决此错误的想法。

谢谢

public class ChildList extends ListFragment implements
    LoaderManager.LoaderCallbacks<Cursor>, DatabaseConstants, GoodiList {

private final String TAG = "ChildList";

// database columns that we will retreive
final String[] PROJECTION = new String[] { CHILD_ID_COLUMN,
        CHILD_NAME_COLUMN };

// selects all
final String SELECTION = null;

SimpleCursorAdapter adapter = null;

private ListParent parentActivity;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    Log.e(TAG, "on Create");
    String[] fromColumns = { CHILD_NAME_COLUMN, CHILD_ID_COLUMN };

    int[] toViews = { R.id.name_column, R.id.id_column };

    adapter=new SimpleCursorAdapter(
            getActivity(),
            R.layout.child_list_entry,
            null,
            fromColumns,
            toViews,
            0
            );

    setListAdapter(adapter);

    getLoaderManager().initLoader(0, null, this);

    Log.i(TAG, "finished on Create");
}


public void onActivityCreated(Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);

    LongClickListener longClickListener = new LongClickListener();

    getListView().setOnItemLongClickListener(longClickListener);
}

@Override
public void setListAdapter(ListAdapter adapter) {
    super.setListAdapter(adapter);
}

@Override
public Loader<Cursor> onCreateLoader(int arg0, Bundle arg1) {

    Log.v(TAG, "starting on create loader");

    Uri CHILD_URI = ChildContentProvider.CONTENT_URI;

    CursorLoader cursorLoader = new CursorLoader(getActivity(), CHILD_URI,
            PROJECTION, SELECTION, null, null);

    Log.v(TAG, "finished on create loader");

    return cursorLoader;
}

@Override
public void onLoadFinished(Loader<Cursor> arg0, Cursor newCursor) {
    adapter.swapCursor(newCursor);

    if (newCursor != null) {
        int rowCount = newCursor.getCount();

        Log.v(TAG, "--- onLoadFinished:  got " + rowCount + " children");
    }
}

/**
 * this used to delete. However, we need it to select usually. Now delete is
 * done by long click.
 */
@Override
public void onListItemClick(ListView l, View v, int position, long id) {
    super.onListItemClick(l, v, position, id);

    // delete child - remove child from list and database
    Object item = l.getItemAtPosition(position);

    Cursor cursor = (Cursor) item;

    int nameCol = cursor.getColumnIndex(CHILD_NAME_COLUMN);
    int idCol = cursor.getColumnIndex(CHILD_ID_COLUMN);

    final String name = cursor.getString(nameCol);

    final int childId = cursor.getInt(idCol);

    Log.i(TAG, "selected " + name + " id: " + childId + " at position "
            + position);

    cursor.close();

    parentActivity.listItemSelected(this, position, childId);

}

/**
 * Forces cursor to requery database and list to be update. Used when a new
 * child is entered in parent activity's EditText field.
 */
public void notifyDataChanged() {

    Log.i(TAG, "told adapter that data changed");

    getLoaderManager().restartLoader(0, null, this);

    adapter.notifyDataSetChanged();
}

/** tell adapter that cursor is no longer valid*/
@Override
public void onLoaderReset(Loader<Cursor> arg0) {
Log.i(TAG,"---- onLoaderReset -----");

adapter.swapCursor(null);

}

@Override
public void onAttach(Activity activity) {
    super.onAttach(activity);

    parentActivity = (ListParent) activity;

    parentActivity.setList(this);

}

【问题讨论】:

  • 这不是错误信息,是警告...!!
  • 虽然应该有一些方法来避免它。
  • @jeff-of-brooklyn 你找到答案了吗?我现在正在努力解决同样的问题。
  • 我已经通过小心地关闭光标来消除它。正如 Igotsar 所说,这是一个警告,因此它不会使我的代码崩溃。

标签: android sqlite android-listfragment


【解决方案1】:

我遇到了同样的问题,我通过仔细查看文档解决了它。

根据文档,http://developer.android.com/reference/android/widget/CursorAdapter.html#swapCursor(android.database.Cursor)

swapCursor(Cursor data) 方法返回原始未关闭的游标。如果新游标与旧游标相同或不存在旧游标,则返回 null。

通过更改我的代码

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

@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
    Cursor cursor = adapter.swapCursor(data);
    if (cursor != null) {
        cursor.close();
    }
}

我阻止了这些警告的出现。

注意:您也应该对 onLoaderReset() 方法执行相同的操作。因为即使您执行adapter.swapCursor(null),它也会返回旧光标。

【讨论】:

    【解决方案2】:

    您应该使用changeCursor(Cursor cursor) 而不是swapCursor(Cursor cursor)

    如果您查看documentation,您会发现它也关闭了旧光标。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-02-14
      • 2019-07-03
      • 2015-06-08
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多