NSURLProtocol解决特定URL使用法与了解

作者 : 开心源码 本文共4406个字,预计阅读时间需要12分钟 发布时间: 2022-05-12 共167人阅读

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不阻拦

//定义一个字符串做keystatic 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子类的对象
这个对象有俩个重要属性使用来解决这个请求

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

总结

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

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

说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » NSURLProtocol解决特定URL使用法与了解

发表回复