NSURLProtocol拦截网络请求用法与理解

NSURLProtocol

官方解释一个抽象类,用于处理特定于协议的URL数据的加载。

使用方法

不要NSURLProtocol直接实例化子类。而是为您的应用支持的任何自定义协议或URL方案创建子类。下载开始时,系统会创建相应的协议对象来处理相应的URL请求。您可以在应用程序的启动时间内定义协议类并调用类方法,以便系统了解您的协议。registerClass:

用NSURLProtocol可以统一处理app内你发的协议,例如你要对请求头进行处理加工,对请求以及响应处理都是个不错的地方

1.首先创建一个继承于NSURLProtocol的类

#import <Foundation/Foundation.h>

@interface XiaDianProtocol : NSURLProtocol

@end

2.然后先在app开启的时候加入如下代码,注册自定义协议类,这样你发的请求都会通过你这类进行过滤在进行下一步操作

[NSURLProtocol registerClass:[XiaDianProtocol class]];
过滤网络请求

3.写一个简单网络请求,API在网上找的看一下效果

  //创建一个网络路径
    NSString *browseUrl = [NSString stringWithFormat:@"https://www.sojson.com/open/api/weather/json.shtml?city=%@", @"北京"];
    //处理一下特殊字符汉字等
    browseUrl =  [browseUrl stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLFragmentAllowedCharacterSet]];
    //创建一个网络请求
    NSURLRequest *request =[NSURLRequest requestWithURL:[NSURL URLWithString:browseUrl]];
    //创建一个Task任务:
    NSURLSession *session = [NSURLSession sharedSession];
    NSURLSessionDataTask *sessionDataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        if (data == nil) {
            return ;
        }
         NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:(NSJSONReadingMutableLeaves) error:nil];
        NSLog(@"从服务器获取到数据%@", dict);
    }];
    NSLog(@"sessionDataTask------>%p", sessionDataTask);
    //执行任务
    [sessionDataTask resume];

4.运行结果当然是崩的 因为protocol里有必须要实现的方法 要不崩至少要实现有下面几个API

方法API 注释
+ (BOOL)canInitWithRequest:(NSURLRequest *)request; 确定协议子类是否可以处理指定的请求。
- (void)startLoading; 启动特定于协议的请求加载。
- (void)stopLoading; 停止特定于协议的请求加载。
+ (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest *)request; 返回指定请求的规范版本。

5.把这四个都实现一下,不要返回值的都可以先空着

// return YES 就是都进行处理抓到就处理
+ (BOOL)canInitWithRequest:(NSURLRequest *)request{
    return YES;
}
//返回规范版本的请求一般直接返回,改变影响查找URL缓存中的对象
+ (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest *)request{
    return request;
}
// 启动特定于协议的请求加载。
- (void)startLoading{
}
// 停止特定于协议的请求加载。
-(void)stopLoading{
}

6.运行发送请求正常顺序就是canInitWithRequest-》canonicalRequestForRequest-》startLoading-》stopLoading

但你会发现你的网络请求都会是超时的,接收不到数据了
因为你拦截了你发的网络请求然后什么也没有做,这个请求就相当于没有发......所以我们要完善一下startloading方法里的东西 这里我们要做的事就是把拦截的请求发出去然后返回到外面的请求回调里

- (void)startLoading{
   //复制一份获取拦截的请求
    NSMutableURLRequest *request = [self.request copy];   
    NSURLSessionDataTask *sessionDataTask = [[NSURLSession sharedSession] dataTaskWithRequest:self.request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
       //将获取的数据回传给外面的请求
        [self.client URLProtocol:self didLoadData:data];
        [self.client URLProtocol:self didReceiveResponse:response cacheStoragePolicy:NSURLCacheStorageNotAllowed];
        [self.client URLProtocol:self didFailWithError:error];
        [self.client URLProtocolDidFinishLoading:self];
    }];
    [sessionDataTask resume];
}

7.运行你会发现一个更大的问题 死循环了 因为你用的是NSURLSessionDataTask发的请求 还会被拦截到 拦截到再发 再拦,所以我们要对我们在startLoading里的请求做一下标识不让它被拦截 原理就是我们在request对象里人为的添加键值进行标识是否被处理了 如果被处理了就在canInitWithRequest方法里返回No不拦截

//定义一个字符串做key
static NSString *xiaDianDealDone = @"xiaDianDealDone";
//修改后的startLoading方法
- (void)startLoading{
    NSMutableURLRequest *request = [self.request copy];
    //为request对象添加一个键值标记为YES
    [NSURLProtocol setProperty:@(YES) forKey:xiaDianDealDone inRequest:request];
    
    NSURLSessionDataTask *sessionDataTask = [[NSURLSession sharedSession] dataTaskWithRequest:self.request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
       //将获取的数据回传给外面的请求
        [self.client URLProtocol:self didLoadData:data];
        [self.client URLProtocol:self didReceiveResponse:response cacheStoragePolicy:NSURLCacheStorageNotAllowed];
        [self.client URLProtocol:self didFailWithError:error];
        [self.client URLProtocolDidFinishLoading:self];
    }];
    [sessionDataTask resume];
}
//处理后的canInitWithRequest方法
+ (BOOL)canInitWithRequest:(NSURLRequest *)request{
   //发现是处理过的请求直接返回No不拦截此请求
    if ([NSURLProtocol propertyForKey:xiaDianDealDone inRequest:request]) {
        return NO;
    }
    return YES;
}

8.这时候运行就可以在外部获得网络数据了 在外面的请求完全看不出来做了什么处理。如果你要对网络请求统一做某些处理的时候就在这个protol中就好了

理解

每个网络请求被拦截的时候系统都会生成一个protocol子类的对象
这个对象有俩个重要属性用来处理这个请求

属性 类型 注释
request NSURLRequest 拦截的请求的request对象有这个对象能获取很多request信息
client id <NSURLProtocolClient> 这个是回调回去重要的属性,每发一个网络请求系统应该都会产生一个client对象来处理网络请求进行回调等操作,而所有的client对象都应该遵守的<NSURLProtocolClient>协议 这样我们通过回调协议的方法就可以把数据以及响应返回最初的请求

总结

1.startLoading 里面随便你用什么再次发送拦截的网络请求 只要能请求就行
2.startLoading里对应的回调方法要回调对应的<NSURLProtocolClient>协议方法,这样就能对应的在外面获取到对应的响应
3.注意死循环发送,加上标识。
4.给予苹果NSURLSession或NSURLConnection的http,https请求可以拦截 如果公司自己实现的应用层协议就不好使了。
5.还有一点多次注册protocol子类会按照后注册的线调用来运行 如果处理的请求就不在像后找,没处理就接着向后寻找处理protocol

NSURLProtocol很强大 在canInitWithRequest和startLoading里能做的事情就有很多,网络请求处理的黑魔法。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • Spring Web MVC Spring Web MVC 是包含在 Spring 框架中的 Web 框架,建立于...
    Hsinwong阅读 22,222评论 1 92
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,244评论 8 265
  • 没有您的世界 我学会了很多 我知道,如何去拼命爱 我知道,有多么羡慕别人的感觉 我知道,生命无常 我知道,生与死的...
    小爪纸阅读 139评论 0 0
  • 你相信每个人一生所经历的挫折是有限的吗?你身边是否有那种前半生过得如狗般的辛苦,后半生却享受着超常人之乐的人。 如...
    爱面包的四夕阅读 220评论 0 4