我正在尝试为Android中的RecyclerView Adapter优化过滤方法。该列表用作ArrayList。我见过这个post但他们每次都会从原始列表中过滤掉。
示例:如果String' a'有10个结果。然后用户输入' m' ,' am'结果是' a'结果(results.size()< = 10)。
我在这个问题上有三点要问,
notifyItemInserted
仍然没有动画)
在下面的代码中,mList用于onBindViewHolder
方法。 copyList始终包含所有数据(不对其进行插入或删除)。
class MyFilter extends Filter {
/**
* 1. check do we have search results available (check map has this key)
* 2. if available, remove all rows and add only those which are value for that key (string)
* 3. else check do we have any key available starting like this, s=har, already available -ha then it can be reused
*
* @param constraint
*/
@Override
protected FilterResults performFiltering(CharSequence constraint) {
//Here you have to implement filtering way
final FilterResults results = new FilterResults();
if (!mSearchMap.containsKey(constraint.toString())) {
String supersetKey = getSupersetIfAvailable(mSearchMap, constraint.toString());
if (supersetKey == null) {
List<Integer> foundPositions = doFullSearch(copyList, constraint.toString());
mSearchMap.put(constraint.toString(), foundPositions);
} else {
List<Integer> foundPositions = filterFromSuperset(copyList, mSearchMap.get(supersetKey), constraint.toString());
mSearchMap.put(constraint.toString(), foundPositions);
}
}
return results;
}
private String getSupersetIfAvailable(Map<String, List<Integer>> mSearchMap, String s) {
Set<String> set = mSearchMap.keySet();
List<String> list = new ArrayList<>(set);
Collections.sort(list);
Collections.reverse(list);
for (String c : list) {
if (s.startsWith(c)) {
return c;
}
}
return null;
}
private List<Integer> filterFromSuperset(List<WeekWorkBean> list, List<Integer> supersetResults, String s) {
List<Integer> results = new ArrayList<>();
String lowerS = s.toLowerCase();
for (int i = 0; i < supersetResults.size(); i++) {
if (list.get(supersetResults.get(i)).getEmpName().toLowerCase().startsWith(lowerS)) {
results.add(supersetResults.get(i));
}
}
return results;
}
private List<Integer> doFullSearch(List<WeekWorkBean> list, String s) {
List<Integer> results = new ArrayList<>();
for (int i = 0; i < list.size(); i++) {
if (list.get(i).getEmpName().toLowerCase().startsWith(s.toLowerCase())) {
results.add(i);
}
}
return results;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
// here you can use result - (f.e. set in in adapter list)
mList.clear();
notifyDataSetChanged();
List<Integer> res = mSearchMap.get(constraint.toString());
int j = 0;
for (Integer i : res) {
mList.add(copyList.get(i));
notifyItemInserted(j++);
}
}
}
答案 0 :(得分:3)
检查一下 https://medium.com/@iammert/using-diffutil-in-android-recyclerview-bdca8e4fbb00#.ehc0gaijt
您正在寻找DiffUtils。您可以在Rx链中使用它将其移出mainThread以获取大数据。 这是一个例子 https://medium.com/@nullthemall/diffutil-is-a-must-797502bc1149#.yg35y9q9b
答案 1 :(得分:1)
要给出第二点的答案,你可以试试这个:
notifyItemRangeChanged(pos, ItemList.size());
答案 2 :(得分:1)
HashMap是一个Map,还有TreeMap,LinkedHashMap和Hashtable。他们每个人都有自己的功能,界面是Map,而不是Collection。您还可以使用其他数据结构,如Treeset,HashSet,ArrayList,LinkedList等。这些结构来自Set和List接口,它扩展到Collection接口。你可以使用它们中的每一个。
如果您向集合中插入任何对象,请使用notifyItemInserted(int position)
,如果删除任何对象,请使用notifyItemRemoved(int position)
,如果您使用notifyDataSetChanged()
更新任何对象。注意收集长度和适配器视图计数的相等性。
您可以在地图中存储您想要的参数。没有任何限制。但是你应该为你选择最好的收藏,设置,列表或地图。