iOS网络请求太频繁 处理之前发送的http请求(取消)

搜索功能在APP中非常的常见,搜索功能伴随的往往是实时搜索结果,极大的方便了用户的查找与实时数据的更新,但是也有极大的问题,当我们搜索框的文字改变的时候,就会进行网络请求,如果输入特别快的时候,网络请求也会特频繁,对服务器的压力也就更大。

解决方法:在进行新的网络请求的时候,把之前的http请求取消掉,保证同一时间只有一个http请求在执行,这样就极大的优化了性能,同时减小了服务器的压力

下面我们用常用的网络请求第三方----- AFNetWorking 为例简单明了的演示一下

大致思路:声明一个全局的 AFHTTPSessionManager *manager;再声明一个全局的可变数组用于存放 http请求。

PS:进行http请求的时候,一定要创建一个新的 NSURLSessionDataTask 任务

再PS:[task cancel] 之后,并不是说这个http请求就完全没有任何音讯了,task会立刻回掉 error,NSLog(@"error:%@",error.description); 可以打印log消息看到 Code=-999 “已取消”

 

上代码

#import "LPDemoVC.h"
 
#import 
 
@interface LPDemoVC ()
 
@property (nonatomic,strong)    AFHTTPSessionManager    *manager;
@property (nonatomic,strong)    NSMutableArray          *taskArray;
 
@end
 
@implementation LPDemoVC
 
#pragma mark -- taskArr
-(NSMutableArray *)taskArray{
    if (!_taskArray) {
        _taskArray = [NSMutableArray arrayWithCapacity:3];
    }
    return _taskArray;
}
 
- (void)viewDidLoad {
    [super viewDidLoad];
    //初始化  AFHTTPSessionManager 并且设置配置信息,在这不做多述
    self.manager = [AFHTTPSessionManager manager];
    self.manager.responseSerializer = [AFHTTPResponseSerializer serializer];
    
}
 
-(BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string{
    //获取当前输入字符串,并且调用网络数据
    NSString *searchStr = [textField.text stringByReplacingCharactersInRange:range withString:string];
    [self loadDataWithSearchStr:searchStr];
    return YES;
}
 
-(void)loadDataWithSearchStr:(NSString *)searchStr{
    //首先遍历存放task的数组,并且取消之前的http请求
    [self.taskArray enumerateObjectsUsingBlock:^(NSURLSessionDataTask *task, NSUInteger idx, BOOL * _Nonnull stop) {
        NSLog(@"这个任务取消之前的状态是:::::%ld",task.state);
        [task cancel];//取消http请求
        NSLog(@"这个任务取消之后的状态是:::::%ld",task.state);
    }];
    //移除self.taskArray中所有的任务
    [self.taskArray removeAllObjects];
    //创建一个新的NSURLSessionDataTask,记住,一定是新创建一个任务
    NSURLSessionDataTask *newTask = [self.manager GET:@"你的接口url" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSDictionary *JSON = [NSJSONSerialization JSONObjectWithData:responseObject options:NSJSONReadingMutableContainers error:nil];
        NSLog(@"success:%@",JSON);
        //在这里处理数据
        
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        NSLog(@"error:%@",error.description);
    }];
    //把最新的任务存进self.taskArray,用于下次遍历
    [self.taskArray addObject:newTask];
}
 
/*
#pragma mark - Navigation
// In a storyboard-based application, you will often want to do a little preparation before navigation
- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
    // Get the new view controller using [segue destinationViewController].
    // Pass the selected object to the new view controller.
}
*/
 
@end


这里给出NSURLSessionTaskState的状态码


typedef NS_ENUM(NSInteger, NSURLSessionTaskState) {
    NSURLSessionTaskStateRunning = 0,                     /* The task is currently being serviced by the session */
    NSURLSessionTaskStateSuspended = 1,
    NSURLSessionTaskStateCanceling = 2,                   /* The task has been told to cancel.  The session will receive a URLSession:task:didCompleteWithError: message. */
    NSURLSessionTaskStateCompleted = 3,                   /* The task has completed and the session will receive no more delegate notifications */
} NS_ENUM_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);


搞定!!!
 

你可能感兴趣的:(iOS网络请求太频繁 处理之前发送的http请求(取消))