3

我对android还很陌生,过滤后的listView存在一些问题,它从横向模式变为纵向模式或反之亦然。我有一个用于过滤“drinkSearch”的editText,只要我不改变视角(纵向与横向),此过滤就可以工作。这是我得到的错误:

java.lang.IllegalStateException:尝试重新打开一个已经关闭的对象:SQLiteQuery:SELECT _id,name FROM Drinks

正如您在下面的代码中看到的那样,我使用了接口 LoaderManager.LoaderCallbacks,这个概念对我来说有点新,我不确定哪里出了问题。我将感谢所有帮助,在此先感谢!

公共类 Drinks_Fragment 扩展 Fragment 实现 LoaderManager.LoaderCallbacks {

private static final int DRINKS_LIST_LOADER = 0x01;
private SimpleCursorAdapter adapter;
private ListView drinksList;
private String LOG;
private EditText drinkSearch;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    View view = inflater.inflate(R.layout.drinks_list, container, false);
    drinkSearch = (EditText)view.findViewById(R.id.drinkInputSearch);
    drinksList = (ListView) view.findViewById(R.id.drinksList);
    drinksList.setEmptyView(view.findViewById(R.id.empty_list_view));

    String[] from = {DrinksTable.COLUMN_NAME};
    int[] to = {R.id.drinkName};
    getLoaderManager().initLoader(DRINKS_LIST_LOADER, null, this);
    adapter = new SimpleCursorAdapter(getActivity().getApplicationContext(), R.layout.drinks_list_item,null, from, to, 0);
    drinksList.setAdapter(adapter);

在这一部分中,我根据在 searchDrink 编辑文本中输入的字符串向我的 contentProvider 请求一个新的光标。(以下代码,直到“返回视图”就在上面部分的下方,相同的 onCreateView 方法)

    drinkSearch.addTextChangedListener(new TextWatcher() {

        @Override
        public void onTextChanged(CharSequence s, int arg1, int arg2, int arg3) {
            // When user changed the Text
            adapter.getFilter().filter(s.toString());
        }

        @Override
        public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) {
            // TODO Auto-generated method stub

        }

        @Override
        public void afterTextChanged(Editable arg0) {
            // TODO Auto-generated method stub
        }
    });

    adapter.setFilterQueryProvider(new FilterQueryProvider() {

        public Cursor runQuery(CharSequence constraint) {
            String value = "%"+constraint.toString()+"%";
            ContentResolver content = getActivity().getContentResolver();
            return content.query(CupProvider.DRINKS_URI,new String[]{DrinksTable.COLUMN_ID,DrinksTable.COLUMN_NAME},DrinksTable.COLUMN_NAME + " LIKE ?",new String[]{value},null);
        }
    });

    return view;
}

@Override
public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {
    super.onCreateContextMenu(menu,v,menuInfo);
    MenuInflater inflater = getActivity().getMenuInflater();
    inflater.inflate(R.menu.drink_actions,menu);
}


@Override
public Loader<Cursor> onCreateLoader(int i, Bundle bundle) {
    String[] projection = {DrinksTable.COLUMN_ID, DrinksTable.COLUMN_NAME};
    CursorLoader cursorLoader = new CursorLoader(getActivity(), CupProvider.DRINKS_URI, projection, null, null, null);
    return cursorLoader;
}

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

}

@Override
public void onLoaderReset(Loader<Cursor> cursorLoader) {
    // data is not available anymore, delete reference
    adapter.swapCursor(null);
}

}

这里有 2 张图片来展示它目前的样子:http://oi42.tinypic.com/dfc702.jpg http://oi43.tinypic.com/2ylqkqa.jpg

4

4 回答 4

6

由于格式不佳,您的代码有点难以理解。

无论如何,提供的答案实际上不是一个修复。应保证返回的光标onLoadFinished不会关闭,因此您以错误的方式加载光标。具体来说,当你打电话

adapter.getFilter().filter(s.toString());

我真的不明白这里发生了什么,但我明白你应该做点别的。只需将查询过滤器存储在您的字段中Fragment并运行getLoaderManager().restartLoader(DRINKS_LIST_LOADER, null, this);. 请注意,您运行的是restartLoader,而不是initLoader,因为您有不同的数据要查询。

在您的onCreateLoader中,您应该使用您作为实例变量存储的过滤器selection

一些背景

initLoader加载上次运行时加载的数据(如果之前运行过)。这就是为什么您在片段/活动的初始化方法中调用。这很方便,因为您不必重新查询方向更改。

restartLoader清理以前加载的数据,以便您获得一个新Loader的来处理(可能)不同的数据。


如果您还不确定自己在做什么,请务必阅读这篇文章,这是一篇关于加载器的非常好的介绍性文章,其中的示例代码看起来非常像您想要实现的目标。装载机起初非常神秘,但一旦你掌握了它的窍门,它就会一帆风顺。

于 2013-10-21T12:14:59.620 回答
3

修复:

@Override public void onLoadFinished(Loader cursorLoader, Cursor cursor) { if(!cursor.isClosed()){ adapter.swapCursor(cursor); } }

我现在确实有另一个问题,我的列表没有自动更新...

于 2013-07-08T11:10:56.437 回答
2

在 AndroidManifest 中添加这一行在同样的情况下帮助了我:

android:configChanges="keyboardHidden|orientation|screenSize"
于 2014-09-18T19:16:15.753 回答
1

这是因为过滤后,原来的游标被关闭了,因为调用了changeCursor。覆盖 changeCursor :

super.swapCursor(cursor)

在您的 SimpleCursorAdapter 中确保原始光标由 CursorLoader 管理。

于 2015-01-22T12:07:40.333 回答