IOS理论知识网络请求iOS开发_性能优化

iOS 如何避免在短时间内频繁发出相同的网络请求?

2016-11-18  本文已影响2152人  August24
  1. UI控制

    这种控制在下拉刷新时体现的比较明显。观察下拉刷新框架时我们会发现下拉触发网络请求后,界面UI会处于一个刷新状态,直到此次的请求成功或失败时界面才会恢复至常态,此时界面才支持下一次的刷新,这样的控制保证了短时间内只有一个请求在运行。


    新浪微博的下拉刷新
  2. 根据需求只保留第一次发出的请求

这种情况多发生在点击一个按钮之后要发送一个请求,而且每次的请求携带的参数是相同的,即每次发送出去的都是同一个请求,此时我们根据需要只保留第一次发送出去的请求即可。

  1. 根据需求只保留最后一次发出的请求

这种情况相对于第二种情况的差别是每次的请求携带的参数是不同的,比如搜索时,在用户的每次输入后都会主动触发请求,用户输入的文字会作为一个参数,如果用户慢速地依次输入c、cc、ccc时我们发现搜索的提示信息是会跟随着输入的内容发生改变的。假设每次输入都会触发网络请求,在用户快速输入的时候,如果对网络请求不做处理时会发生这样一种情况,因为请求是异步的,很有可能我们最后展示的网络返回信息并不是最后一次发送出的请求返回来的。这种情况我们就要根据需求只保留最后一次发出的请求即可。

输入c
输入cc

基于AFNetworking框架,我找到了实现情况2、3的方法。先看效果

for (int i = 0; i < 20; i++) {
    [Weather loadWeatherInformationWithCallBack:^(NSArray *array, NSError *error) {
        if (error) {
            NSLog(@"%@", error);
            return;
        } else {
            self.testArray = [self.testArray arrayByAddingObjectsFromArray:array];
            self.weather = self.testArray;
            NSLog(@"-------------------------%d", i);
        }
    } keepFirst:NO];
 }

我们在控制器中用for循环发出20个请求,如果keepFirst参数为NO,控制台输出为:

保留最后一次请求

如果keepFirst参数为YES,控制台输出为:

保留第一次请求

网络单例中的代码如下:

// 加载天气信息,并以keepFirst参数来标示是否保留第一次的请求
- (void)loadWeatherInformationWithCallBack:(callBack)callBack keepFirst:(BOOL)isKeepFirst {
    NSString *string = @"data/sk/101180101.html";
    [self requestMethod:@"GET" URLString:string params:nil finish:callBack keepFirst:isKeepFirst];
}

// get和post的封装
- (void)requestMethod:(NSString *)method URLString:(NSString *)string params:(NSDictionary *)params finish:(callBack)callBack keepFirst:(BOOL)isKeepFirst {
    
    // 成功回调
    void(^successBlock)(NSURLSessionDataTask *, id) = ^(NSURLSessionDataTask *task, id responseObject) {
        // 清除缓存
        [self.datas removeObjectForKey:string];
        if (responseObject) {
            callBack(responseObject, nil);
        } 
    };
    // 失败回调
    void(^failBlock)(NSURLSessionDataTask *, NSError *) = ^(NSURLSessionDataTask *task, NSError *error) {
        // 清除缓存
        [self.datas removeObjectForKey:string];

        // 如果是我们主动调用  [originalTask cancel]; 的话会立即返回到失败的回调中,且error中的一个字段会被标示为`cancelled`,我们以此来过滤掉取消的task的回调。
        if ([error.userInfo[@"NSLocalizedDescription"] isEqualToString:@"cancelled"]) {
            return;
        }
        callBack(nil, error);
    };
    
    NSURLSessionDataTask *originalTask = [self.datas objectForKey:string];

    // 如果保留第一次请求,且存在初始请求,return
    if (isKeepFirst && originalTask) {
        return;
    }
    
    [originalTask cancel];
    NSURLSessionDataTask *task = nil;
    if ([method isEqualToString:@"GET"]) {
        task = [self GET:string parameters:params success:successBlock failure:failBlock];
    }
    if ([method isEqualToString:@"POST"]) {
        task = [self POST:string parameters:params success:successBlock failure:failBlock];
    }
    // 网络单例中会用一个字典属性来存储网络请求,为加以区分以每个请求的`url` 做为 key 进行存储
    [self.datas setValue:task forKey:string];
}

完整的项目已经上传到github上了项目地址

上一篇 下一篇

猜你喜欢

热点阅读