Android SearchView:可过滤的 Listview 运行时错误
Android SearchView: Filterable Listview Runtime Error
我 运行 在使用可过滤列表视图时偶尔会遇到问题。在过滤食谱列表视图的同时从搜索视图中删除文本时,我很少会遇到运行时错误。当使用软键盘快速删除文本时,往往会发生这种情况。
90% 的时间,这个实现完全符合我的预期,但是当快速删除字符时,我收到以下错误。
04-07 11:38:55.221 9591-9591/brd.cms.sup E/dalvikvm﹕ >>>>> Normal User
04-07 11:38:55.221 9591-9591/brd.cms.sup E/dalvikvm﹕ >>>>> brd.cms.sup [ userId:0 | appId:10359 ]
04-07 11:38:59.295 9591-9591/brd.cms.sup E/OpenGLRenderer﹕ GL_INVALID_OPERATION
04-07 11:40:58.018 9591-9591/brd.cms.sup E/AndroidRuntime﹕ FATAL EXCEPTION: main
Process: brd.cms.sup , PID: 9591
java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. Make sure your adapter calls notifyDataSetChanged() when its content changes. [in ListView(2131493035, class android.widget.ListView) with Adapter(class brd.cms.sup.RecipeAdapter)]
at android.widget.ListView.layoutChildren(ListView.java:1566)
at android.widget.AbsListView.onLayout(AbsListView.java:2598)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.support.v4.widget.DrawerLayout.onLayout(DrawerLayout.java:890)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.support.v7.internal.widget.ActionBarOverlayLayout.onLayout(ActionBarOverlayLayout.java:502)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2379)
at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2092)
at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1267)
at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6640)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:813)
at android.view.Choreographer.doCallbacks(Choreographer.java:613)
at android.view.Choreographer.doFrame(Choreographer.java:583)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:799)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:146)
at android.app.ActivityThread.main(ActivityThread.java:5635)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1291)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1107)
at dalvik.system.NativeStart.main(Native Method)
下面是我的 SearchView.OnQueryTextListener 实现。
@Override
public boolean onQueryTextChange(String newText) {
if (recipeAdapter != null) {
try {
recipeAdapter.getFilter().filter(newText);
}
catch (Exception e) {
e.printStackTrace();
}
}
return false;
}
这是 RecipeAdapter 中我的私有 RecipeFilter。
private class RecipeFilter extends Filter{
List<Recipe> filteredList = new ArrayList<>();
@Override
protected FilterResults performFiltering(CharSequence constraint){
constraint = constraint.toString().toLowerCase();
FilterResults result = new FilterResults();
filteredList.clear();
if (constraint != null && constraint.toString().length() > 0) {
for (Recipe r : backupList) {
if (r.contains(constraint)) {
filteredList.add(r);
}
}
} else {
filteredList.addAll(backupList);
}
result.count = filteredList.size();
result.values = filteredList;
return result;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
recipeList = (List) results.values;
if (results != null) {
notifyDataSetChanged();
}
else{
notifyDataSetInvalidated();
}
}
}
如有任何想法,我们将不胜感激!
了解如何使用 filteredList
和 recipeList
会有所帮助。但是,您的 ListView
不同步可能与缺乏同步有关。 performFiltering()
方法在后台线程上执行。您必须同步那些 Lists
上的任何突变...不仅在 performFiltering()
方法内,而且在整个适配器中。顺便说一句,我当然希望您的自定义适配器不是 ArrayAdapter
的扩展。这本身就是另一个问题。
这里有一篇 short blog post 讨论了 Android 的 ArrayAdapter
的过滤问题。在它的最后,您会发现示例代码显示了为适配器编写 Filter
class 的好方法。基本上与您在此处发布的等效代码。
管理列表
总而言之,您有一个 recipeList
,它包含适配器中的所有项目或仅包含已过滤的项目。您还有一个 backupList
存储所有项目。然后在Filter
class里面有一个filterList
,用来方便过滤后的数据变成recipeList
.
这种方法是相当标准的...其中两个主要列表用于跟踪过滤后的数据和原始数据。使用这些列表的关键是同步并知道要使用哪一个。
对于任何 getter,您的适配器应始终处理来自 recipeList
的数据。例如,getItem()
、getItemId()
等。访问此数据不需要同步。
对于任何 setter,您将执行以下操作
- 同步
- 确定更新哪个列表
- 致电
notifyDataSetChanged
关于 #2,根据您的设置,无论您的 backupList
是什么,都应该更新。您的 recipeList
是否也得到更新取决于您是否希望在过滤期间应用突变。
最后,您的过滤器 class 需要添加同步。特别是当您将数据从 backupList
复制到 filteredList
时。此外,filteredList
需要对方法 performFiltering()
是局部的,而不是对 class 是全局的。或者同步整个方法。由于该方法发生在后台线程上,因此多个线程很可能同时执行该方法。这意味着他们都在改变相同的 filteredList
...这并不好。
此外,在 publishResults
中,您需要同步更新 recipeList
并且需要在结果为 null 或为空时调用 notifyDataSetInvalidated()
。
我 运行 在使用可过滤列表视图时偶尔会遇到问题。在过滤食谱列表视图的同时从搜索视图中删除文本时,我很少会遇到运行时错误。当使用软键盘快速删除文本时,往往会发生这种情况。
90% 的时间,这个实现完全符合我的预期,但是当快速删除字符时,我收到以下错误。
04-07 11:38:55.221 9591-9591/brd.cms.sup E/dalvikvm﹕ >>>>> Normal User
04-07 11:38:55.221 9591-9591/brd.cms.sup E/dalvikvm﹕ >>>>> brd.cms.sup [ userId:0 | appId:10359 ]
04-07 11:38:59.295 9591-9591/brd.cms.sup E/OpenGLRenderer﹕ GL_INVALID_OPERATION
04-07 11:40:58.018 9591-9591/brd.cms.sup E/AndroidRuntime﹕ FATAL EXCEPTION: main
Process: brd.cms.sup , PID: 9591
java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. Make sure your adapter calls notifyDataSetChanged() when its content changes. [in ListView(2131493035, class android.widget.ListView) with Adapter(class brd.cms.sup.RecipeAdapter)]
at android.widget.ListView.layoutChildren(ListView.java:1566)
at android.widget.AbsListView.onLayout(AbsListView.java:2598)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.support.v4.widget.DrawerLayout.onLayout(DrawerLayout.java:890)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.support.v7.internal.widget.ActionBarOverlayLayout.onLayout(ActionBarOverlayLayout.java:502)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
at android.view.View.layout(View.java:15860)
at android.view.ViewGroup.layout(ViewGroup.java:4902)
at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2379)
at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2092)
at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1267)
at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6640)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:813)
at android.view.Choreographer.doCallbacks(Choreographer.java:613)
at android.view.Choreographer.doFrame(Choreographer.java:583)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:799)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:146)
at android.app.ActivityThread.main(ActivityThread.java:5635)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1291)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1107)
at dalvik.system.NativeStart.main(Native Method)
下面是我的 SearchView.OnQueryTextListener 实现。
@Override
public boolean onQueryTextChange(String newText) {
if (recipeAdapter != null) {
try {
recipeAdapter.getFilter().filter(newText);
}
catch (Exception e) {
e.printStackTrace();
}
}
return false;
}
这是 RecipeAdapter 中我的私有 RecipeFilter。
private class RecipeFilter extends Filter{
List<Recipe> filteredList = new ArrayList<>();
@Override
protected FilterResults performFiltering(CharSequence constraint){
constraint = constraint.toString().toLowerCase();
FilterResults result = new FilterResults();
filteredList.clear();
if (constraint != null && constraint.toString().length() > 0) {
for (Recipe r : backupList) {
if (r.contains(constraint)) {
filteredList.add(r);
}
}
} else {
filteredList.addAll(backupList);
}
result.count = filteredList.size();
result.values = filteredList;
return result;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
recipeList = (List) results.values;
if (results != null) {
notifyDataSetChanged();
}
else{
notifyDataSetInvalidated();
}
}
}
如有任何想法,我们将不胜感激!
了解如何使用 filteredList
和 recipeList
会有所帮助。但是,您的 ListView
不同步可能与缺乏同步有关。 performFiltering()
方法在后台线程上执行。您必须同步那些 Lists
上的任何突变...不仅在 performFiltering()
方法内,而且在整个适配器中。顺便说一句,我当然希望您的自定义适配器不是 ArrayAdapter
的扩展。这本身就是另一个问题。
这里有一篇 short blog post 讨论了 Android 的 ArrayAdapter
的过滤问题。在它的最后,您会发现示例代码显示了为适配器编写 Filter
class 的好方法。基本上与您在此处发布的等效代码。
管理列表
总而言之,您有一个 recipeList
,它包含适配器中的所有项目或仅包含已过滤的项目。您还有一个 backupList
存储所有项目。然后在Filter
class里面有一个filterList
,用来方便过滤后的数据变成recipeList
.
这种方法是相当标准的...其中两个主要列表用于跟踪过滤后的数据和原始数据。使用这些列表的关键是同步并知道要使用哪一个。
对于任何 getter,您的适配器应始终处理来自 recipeList
的数据。例如,getItem()
、getItemId()
等。访问此数据不需要同步。
对于任何 setter,您将执行以下操作
- 同步
- 确定更新哪个列表
- 致电
notifyDataSetChanged
关于 #2,根据您的设置,无论您的 backupList
是什么,都应该更新。您的 recipeList
是否也得到更新取决于您是否希望在过滤期间应用突变。
最后,您的过滤器 class 需要添加同步。特别是当您将数据从 backupList
复制到 filteredList
时。此外,filteredList
需要对方法 performFiltering()
是局部的,而不是对 class 是全局的。或者同步整个方法。由于该方法发生在后台线程上,因此多个线程很可能同时执行该方法。这意味着他们都在改变相同的 filteredList
...这并不好。
此外,在 publishResults
中,您需要同步更新 recipeList
并且需要在结果为 null 或为空时调用 notifyDataSetInvalidated()
。