项目中很多地方需要搜索, 每次搜索都是调用服务端提供的接口 . 对这些事情做个总结吧.
Q:
需求如下 ,
用户点击搜索栏,跳出键盘并输入,每次输入完毕, 用另一个tableview展示搜索结果.
分析:
在iOS中无论是哪一种搜索, 似乎直接用系统的库UISearchbar是最简单的.
- (void)searchBar:(UISearchBar *)searchBar textDidChange:(NSString *)searchText
利用它的回调方法进行搜索. 看上去真的很简单. 哪怕控件麻烦一点,用UITextfield这也不会是问题 .
坑来了. 当每次text变化, 都会触发这个方法 . 如果我在这里发送请求, 访问服务端, 会出现一个问题.
通常用户打字比较快, 会疯狂的输入,这里就会疯狂的触发请求,导致流量,内存过渡的浪费了, 性能大大降低, 要重复请求这么多次, 这肯定是不合理的####
A:
解决思路
把这些请求NSURLSessionDataTask
放到一个队列里, 每次执行textDidChange
方法, 先对这个队列进行处理, cancel
之前的请求.
上代码
#import <Foundation/Foundation.h>
typedef void(^BlockSearchTaskComplete)(NSURLSessionDataTask *task, id responseObject) ;
@interface XTSearchHandler : NSObject
@property (nonatomic,copy) BlockSearchTaskComplete searchComplete ;
- (void)searchWithText:(NSString *)searchText ;
@end
@interface XTSearchHandler ()
@property (nonatomic,strong) NSMutableArray *arrayOfTasks ;
@property (nonatomic,strong) AFHTTPSessionManager *manager ;
@end
@implementation XTSearchHandler
- (instancetype)init
{
self = [super init];
if (self) {
self.manager = [[AFHTTPSessionManager alloc] init] ;
self.manager.responseSerializer = [AFJSONResponseSerializer serializer] ;
self.arrayOfTasks = [NSMutableArray new] ;
}
return self;
}
- (void)searchWithText:(NSString *)searchText
{
if (searchText.length >= 2)
{
/// cancel all previous tasks
[self.arrayOfTasks enumerateObjectsUsingBlock:^(NSURLSessionDataTask *taskObj, NSUInteger idx, BOOL *stop) {
[taskObj cancel] ; /// when sending cancel to the task failure: block is going to be called
}];
/// empty the arraOfTasks
[self.arrayOfTasks removeAllObjects];
/// init new task
NSURLSessionDataTask *task = [ServerRequest searchArticleTagWithSearchKey:searchText
manager:self.manager
success:^(NSURLSessionDataTask *task, id responseObject) {
self.searchComplete(task,responseObject) ;
} fail:^(NSURLSessionDataTask *task, NSError *error) {
}] ;
/// add the task to our arrayOfTasks
[self.arrayOfTasks addObject:task] ;
}
}
@end