一、简介

一般来说,我们可以使用以下几种方式实现搜索:

1)暴力搜索——直接使用数据库提供的功能,每次都从数据库中读取搜索的结果,存进一个数据结构用于Adapter显示,调用notifyDataSetChanged()刷新数据;

2)利用filter进行搜索。这块涉及到Filterable接口。推荐文章:Android实现Filterable通过输入文本框实现联系人自动筛选。有两点值得注意:a)Android原生组件AutoCompleteTextview就是使用该方法实现的;b)这个方法本质上还是调用notifyDataSetChanged()方法,并且还是要自己去实现搜索部分,只是整个方法看上去比较优雅,而且不用再去搜索数据库,最重要的一点是这个时候搜索过程被自动移到另外一个线程之中,搜索完毕之后才会刷新UI;

二、Android做了哪些? 1、为了实现数据的过滤,andorid设计了抽象类Filter,进行异步和同步的数据过滤操作。 2、在Adapter中继承Filterable,提供给使用者Filter,进行过滤。 3、在不同的View中,获得查询约束字符串,传递给Adapter,并且提供配合数据过滤的界面支持。 三、Filter类 Filter的使用流程如下: 调用filter方法 ->在另一线程中调用performFiltering进行数据查询->得到数据过滤结果后调用publishResults将结果返回到使用它的客户端。 该类中的performFiltering和publishResults均为抽象方法,需要继承者自己重写。 比如我们可以自定义一个Filter类实现performFiltering和publishResults方法就可以实现过滤了。

CursorFilter类就是Filter类的继承。CursorFilter在performFiltering中并没有直接进行数据的过滤,而是加入了CursorFilterClient成员,将过滤的操作转让给了CursorFilterClient,实际上CursorAdpater就是继承了CursorFilterClient接口,也就是说过滤操作实际上是在CursorAdapter中执行的。

CursorAdapter实现Filterable接口实现了下面的方法

public Filter getFilter() {
if (mCursorFilter == null) {
mCursorFilter = new CursorFilter(this);
}
return mCursorFilter;
}

看一下CursorFilter的实现,实现Filter的两个抽象方法

class CursorFilter extends Filter {
CursorFilterClient mClient;

CursorFilter(CursorFilterClient client) {
mClient = client;
}

@Override
protected FilterResults performFiltering(CharSequence constraint) {

Cursor cursor = mClient.runQueryOnBackgroundThread(constraint);
FilterResults results = new FilterResults();
if (cursor != null) {
results.count = cursor.getCount();
results.values = cursor;
} else {
results.count = 0;
results.values = null;
}
return results;
}

@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
Cursor oldCursor = mClient.getCursor();
if (results.values != null && results.values != oldCursor) {
mClient.changeCursor((Cursor) results.values);
}
}

查询工作交给了CursorFilterClient这个类,而CursorAdapter实现了这个接口,所以

Cursor cursor = mClient.runQueryOnBackgroundThread(constraint);

这个方法是在CursorAdapter中实现的,我们看下

public Cursor runQueryOnBackgroundThread(CharSequence constraint) {
if (mFilterQueryProvider != null) {
return mFilterQueryProvider.runQuery(constraint);
}
return mCursor;
}

其中的mFilterQueryProvider需要我设置一下

public void setFilterQueryProvider(FilterQueryProvider filterQueryProvider) {
mFilterQueryProvider = filterQueryProvider;
}

看一下FilterQueryProvider这个接口,只有一个抽象方法

public interface FilterQueryProvider {
Cursor runQuery(CharSequence constraint);
}

所以我们需要在外面继承这个接口实现它,并且实现runQuery方法中的查询搜索。

然后通过CursorAdapter的setFilterQueryProvider方法设置,之后整个机制完全串联起来,在外调用adapter.getFilter().filter(et_filter.getText().toString()); 即可启动整个过滤机制。

整个机制实际上是用异步查询然后显示的流程,主要的流程处理在Filter的类中。下篇文章分析下Filter的具体实现

更多相关文章

  1. Android(安卓)kotlin之对象和类(2)
  2. 五、android中解析xml
  3. Android(安卓)Scroll 详解
  4. android launcher开发之图标背景以及默认配置
  5. Android(安卓)沉浸状态栏
  6. RTFSC系列之Retrofit
  7. Android(安卓)APK文件拆解方法
  8. Android(安卓)数据操作(一) 自定义AttributeSet属性
  9. Android(安卓)小项目之--解析如何获取SDCard 内存

随机推荐

  1. Android使用selector改变文字的颜色
  2. android中ProgressBar的使用SeekBar的使
  3. Android 可選文件格式瀏覽器
  4. Android xposed Hook 初探01
  5. Error string types not allowed at andr
  6. Android(Java):长时间未登录提醒
  7. android webview 添加内置对象
  8. android截屏简单引用
  9. Android 获取AndroidManifest.xml 中 met
  10. android HorizontalScrollView