iOS中的两种搜索方式UISearchDisplayController和UISearchController
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处.
如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;)
以前iOS的搜索一般都使用UISearchDisplayController来完成,不过自从iOS 8.0开始,该控制器被标记为废弃,我们可以在iOS 8.0之后使用一个新的搜索控制器UISearchController来完成搜索.
本篇博文将介绍以上2种搜索控制器的简单用法,并比较它们的区别.Let’t Go!
UISearchDisplayController
使用该控制器我们必须给根控制器添加一个UISearchDisplayDelegate协议,然后完成其规定的几个回调方法:
-(BOOL)searchDisplayController:(UISearchDisplayController *)controller
shouldReloadTableForSearchString:(NSString *)searchString{
[_filteredNames removeAllObjects];
if (searchString.length > 0) {
NSPredicate *predicate = [NSPredicate predicateWithBlock:
^BOOL(NSString *name,NSDictionary *b) {
NSRange range = [name rangeOfString:searchString options:NSCaseInsensitiveSearch];
return range.location != NSNotFound;
}];
for (NSString *key in _keys) {
NSArray *matchs = [_names[key] filteredArrayUsingPredicate:predicate];
[_filteredNames addObjectsFromArray:matchs];
}
}
return YES;
}
-(void)searchDisplayController:(UISearchDisplayController *)controller
didLoadSearchResultsTableView:(UITableView *)tableView{
[tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:SectionsTableID];
}
其中第一个方法在搜索关键字发生改变时调用,后一个则在搜索表视图加载完毕后调用,我们可以看见如上代码所示,我们在搜索表视图加载后将其表视图的UITableViewCell元素的ID和一个特定的ID绑定,这样做可以重用已存在的单元格.另外需要注意的是该ID和根视图控制器中标示图的单元格ID是一样的,虽然这两个表视图是不一样的.
我们在根视图控制器的didLoadView中用以下代码初始化UISearchDisplayController:
UISearchBar *searchBar = [[UISearchBar alloc]initWithFrame:CGRectMake(0, 0, 320, 44)];
tableView.tableHeaderView = searchBar;
_searchController = [[UISearchDisplayController alloc]initWithSearchBar:searchBar contentsController:self];
_searchController.delegate = self;
_searchController.searchResultsDataSource = self;
以上代码首先创建一个searchBar并将其设置为tableView的表头视图,接着以该searchBar为参数创建UISearchDisplayController对象,最后设置好其对应的委托.
因为此时根视图中存在2个表视图,所以我们在处理相关回调方法时要区分当前处理的是哪张表视图:
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
if (tableView.tag == 1) {
return _keys.count;
}else{
return 1;
}
}
我们将默认的呈现数据的表视图的tag设置为1,这样我们就知道当前需要处理的是哪张表视图了.其它的回调与此类似,不再一一述及.
UISearchController
下面我们来看看iOS 8.0之后新的搜索器类的使用方法.首先我们也必须添加一个UISearchResultsUpdating协议,同时实现一个协议方法:
-(void)updateSearchResultsForSearchController:(UISearchController *)searchController;
该方法和前一个搜索器类的方法类似,不过要注意的是它不返回值,这意味着如果你想要显示搜索结果,你必须手动刷新表视图.下面给出一个该方法的例子:
-(void)updateSearchResultsForSearchController:(UISearchController *)searchController{
NSString *searchString = _searchController.searchBar.text;
[_filteredNames removeAllObjects];
if (searchString.length > 0) {
NSPredicate *predicate = [NSPredicate predicateWithBlock:
^BOOL(NSString *name,NSDictionary *b) {
NSRange range = [name rangeOfString:searchString options:NSCaseInsensitiveSearch];
return range.location != NSNotFound;
}];
for (NSString *key in _keys) {
NSArray *matchs = [_names[key] filteredArrayUsingPredicate:predicate];
[_filteredNames addObjectsFromArray:matchs];
}
}else{
//如果搜索关键字为空则显示所有key
for (NSString *key in _keys) {
NSArray *matchs = _names[key];
[_filteredNames addObjectsFromArray:matchs];
}
}
UITableView *tableView = [self.view viewWithTag:1];
//手动刷新表视图
[tableView reloadData];
}
UISearchController还有一个和UISearchDisplayController不同的地方,就是判断当前表视图是否是搜索表视图的方法:
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
if (_searchController.active) {
return 1;
}else{
return _keys.count;
}
}
好吧,我收回上面的话…准确来说,应该是前者根本没有自己的表视图,它使用的表视图就是默认的表视图.也就是我们必须自默认的表视图中判断当前搜索是否激活,如果是则显示搜索后的结果,否则显示默认呈现的数据.
最后我们来看看,如何初始化UISearchController:
_searchController = [[UISearchController alloc]initWithSearchResultsController:nil];
_searchController.searchResultsUpdater = self;
_searchController.dimsBackgroundDuringPresentation = NO;
_searchController.hidesNavigationBarDuringPresentation = NO;
tableView.tableHeaderView = _searchController.searchBar;
大家可以看到不用再自己创建UISearchBar对象了,UISearchDisplayController已经帮你搞定了.
总结
最后我们来看一下以上2种搜索控制器的不同:
- 前者使用单独的表视图,后者没有表视图,其内容使用默认的表视图来呈现.
- 前者需要手动创建UISearchBar对象,而后者不需要
- 前者需要在表示图的协议回调中判断当前是哪个表视图,而后者只是简单判断搜索是否处在激活状态
- 它们遵守的协议不一样,协议方法自然也不一样
- 前者比较复杂,后者更简单,从类的构成上来说也更合理.
iOS中的两种搜索方式UISearchDisplayController和UISearchController的更多相关文章
- iOS中的两种主要架构及其优缺点浅析
凡是程序的开发者,应该对程序的架构都不陌生.一个程序的架构的好坏对这个程序有着非常重要的作用.今天我们来看一下iOS开发中用要的两种主流的程序架构.这个过程中我们主要以例子的形式展开. 我们来看第一种 ...
- Javaweb学习笔记——(六)——————xml中jaxp两种解析方式和dom4j运用
1.xml的scheam约束 dtd的语法:<!ElEMENT 元素名称 约束> **schema符合xml的语法,xml语句 **一个xml中可以有多个schema,多个schema使用 ...
- javascript中对象两种创建方式
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- String中的两种实例化方式的区别
直接赋值:(String str = "字符串");只会开辟一块堆内存空间,并且会自动保存在对象池中以供下次重复使用. 构造方法:(String str = new String ...
- web 开发之js---ajax 中的两种提交方式ajax post 和 ajax get 实例
()post http://04101334.iteye.com/blog/637695/ ()get function serializeElement(element) { var method ...
- class类名在webpack项目中的两种引用方式
一.问题描述 在项目工程中,我们通常既用到css module,也用到普通的less文件引用方式,代码及webpack配置如下,运行时,发现只有css module起作用,如何让两者都起作用呢? // ...
- Hibeernate中的两种分页方式
1. return getHibernateTemplate().executeFind(new HibernateCallback() { public Object doInHibernate(S ...
- ios中的界面跳转方式
ios中,两种界面跳转方式 1.NavgationController本身可以作为普通ViewController的容器,它有装Controller的栈,所以可以push和pop它们,实现你所说的跳转 ...
- hibernate 一对一 one to one的两种配置方式
hibernate中one-to-one两种配置方式 标签: hibernateHibernateone-to-one 2013-02-19 17:44 11445人阅读 评论(1) 收藏 举报 分 ...
随机推荐
- 【bzoj4567 scoi2016】 背单词
题目描述 Lweb 面对如山的英语单词,陷入了深深的沉思,”我怎么样才能快点学完,然后去玩三国杀呢?“.这时候睿智的凤老师从远处飘来,他送给了 Lweb 一本计划册和一大缸泡椒,他的计划册是长这样的: ...
- C++符号优先级
C++符号优先级 优先级 操作符 功能 用法 结合性 1 ()[]->.::++-- Grouping operatorArray accessMember access from a poin ...
- php+xdebug远程调试(单人)
目录 服务器上安装 XDebug 及配置 客户端 PHPstorm 配置 浏览器安装插件 服务器上安装 XDebug 及配置 XDebug 安装 略 配置: 打开 php.ini 配置文件: vim ...
- React .js框架的环境搭建
React学习笔记(一)- 环境搭建 最近在学习react相关的知识,刚刚起步,一路遇坑不断.自己做个笔记,方便日后总结,也供相同趣味的小伙伴一起交流探讨. 学习时主要参考官网的教程:https: ...
- CRM客户关系管理系统(一)
第一章.CRM介绍和开发流程 1.1.CRM简介 客户关系管理(CRM) 客户关系管理(customer relationship management)的定义是:企业为提高核心竞争力,利用相应的信息 ...
- Web缓存(一) - HTTP协议缓存
为什么要使用 Web 缓存 Web缓存一般分为浏览器缓存.代理服务器缓存以及网关缓存,本文主要讲的是 浏览器缓存,其它两种缓存大家自行去了解下. Web 缓存游走于服务器和客户端之间.这个服务器可能是 ...
- JDBC:从原理到应用
一.是为何物 1.概念 JDBC(Java Data Base Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用J ...
- Oracle12c中SQL性能优化(SQL TUNING)新特性之自动重优化(automatic reoptimization)
Oracle12c中的自动重优化 Oracle12c中的自适应查询优化有一系列不同特点组成.像自适应计划(AdaptivePlans)功能可以在运行时修改执行计划,但并不允许计划中连接顺序的改变.自动 ...
- vue 2.0 scopedSlots和slots在render函数中的应用示例
渲染内容为: hello from functional render scopedSlots render scopedSlots named slot of render hello from f ...
- Querying CRM data with LINQ
http://www.powerxrm.com/querying-crm-data-with-linq/ 如果不喜欢看SDK中的示例,这篇里面讲的非常详细,值得一看.