大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处.

如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! 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种搜索控制器的不同:

  1. 前者使用单独的表视图,后者没有表视图,其内容使用默认的表视图来呈现.
  2. 前者需要手动创建UISearchBar对象,而后者不需要
  3. 前者需要在表示图的协议回调中判断当前是哪个表视图,而后者只是简单判断搜索是否处在激活状态
  4. 它们遵守的协议不一样,协议方法自然也不一样
  5. 前者比较复杂,后者更简单,从类的构成上来说也更合理.

iOS中的两种搜索方式UISearchDisplayController和UISearchController的更多相关文章

  1. iOS中的两种主要架构及其优缺点浅析

    凡是程序的开发者,应该对程序的架构都不陌生.一个程序的架构的好坏对这个程序有着非常重要的作用.今天我们来看一下iOS开发中用要的两种主流的程序架构.这个过程中我们主要以例子的形式展开. 我们来看第一种 ...

  2. Javaweb学习笔记——(六)——————xml中jaxp两种解析方式和dom4j运用

    1.xml的scheam约束 dtd的语法:<!ElEMENT 元素名称 约束> **schema符合xml的语法,xml语句 **一个xml中可以有多个schema,多个schema使用 ...

  3. javascript中对象两种创建方式

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. String中的两种实例化方式的区别

    直接赋值:(String str = "字符串");只会开辟一块堆内存空间,并且会自动保存在对象池中以供下次重复使用. 构造方法:(String str = new  String ...

  5. web 开发之js---ajax 中的两种提交方式ajax post 和 ajax get 实例

    ()post http://04101334.iteye.com/blog/637695/ ()get function serializeElement(element) { var method ...

  6. class类名在webpack项目中的两种引用方式

    一.问题描述 在项目工程中,我们通常既用到css module,也用到普通的less文件引用方式,代码及webpack配置如下,运行时,发现只有css module起作用,如何让两者都起作用呢? // ...

  7. Hibeernate中的两种分页方式

    1. return getHibernateTemplate().executeFind(new HibernateCallback() { public Object doInHibernate(S ...

  8. ios中的界面跳转方式

    ios中,两种界面跳转方式 1.NavgationController本身可以作为普通ViewController的容器,它有装Controller的栈,所以可以push和pop它们,实现你所说的跳转 ...

  9. hibernate 一对一 one to one的两种配置方式

    hibernate中one-to-one两种配置方式 标签: hibernateHibernateone-to-one 2013-02-19 17:44 11445人阅读 评论(1) 收藏 举报  分 ...

随机推荐

  1. hdu 5012(bfs)

    题意:给你2个 骰子,让你通过翻转使第一个变成第二个,求最少翻转数 思路:bfs #include<cstdio> #include<iostream> #include< ...

  2. BZOJ1095(动态点分治+堆)

    终于把这个坑填了.. 按重心分治建树,每个点存两个堆,第一个存的是这个点子树中的点到父重心的距离,第二个存的是子节点第一个堆的堆顶,同时有一个全局答案堆,存的是每个点第二个堆的最大值+次大值. 20亿 ...

  3. linux内核input子系统解析【转】

    转自:http://emb.hqyj.com/Column/Column289.htm 时间:2017-01-04作者:华清远见 Android.X windows.qt等众多应用对于linux系统中 ...

  4. 零开始:NetCore项目权限管理系统:基础框架搭建

    有兴趣的同学可以一起做 喜欢NetCore的朋友,欢迎加群QQ:86594082 源码地址:https://github.com/feiyit/SoaProJect 新建一个空的解决方案,建立对应的解 ...

  5. ubuntu 系统应用安装方式

    转载   ubuntu系统软件安装 来源:http://www.linuxidc.com/Linux/2011-02/32211.htm 首先说明一下 ubuntu 的软件安装大概有几种方式:1. d ...

  6. C++函数式编程实现牛顿法

    In numerical analysis, Newton's method (also known as the Newton–Raphson method), named after Isaac ...

  7. C语言程序设计 第一次作业

    (一)实验总结 1.求圆面积和周长 (1)题目 输入圆的半径,计算圆的周长和面积. (2)流程图 (3)测试数据及运行结果 (4)实验分析问题一:输出时往输出框输不上.原因 :没有加双引号.2.判断闰 ...

  8. 解决nodejs中json序列化时Date类型为UTC格式

    在nodejs中,json序列化时Date类型时,默认转为UTC格式. 如下图 zhupengfei@DESKTOP-HJASOE3 MINGW64 /d/MyProject/exp2 $ node ...

  9. 虚拟机搭建Zookeeper服务器集群完整笔记

    虚拟机搭建Zookeeper服务器集群完整笔记 本笔记主要记录自己搭建Zookeeper服务器的全过程,默认已经安装部署好Centos7. 一.虚拟机下Centos无法联网解决方案 1.首先调整虚拟机 ...

  10. java中的构造,封装

    今天给大家讲一下面向对象中的构造,封装: 构造:构造方法有以下几个特点:1.方法名和类名一致.2.无返回类型.接下来的几种构造样式,直接上代码吧: //这是一个宠物类 有一个属性:名字(name) p ...