An NSURLProtocol
object handles the loading of protocol-specific URL data. The NSURLProtocol
class itself is an abstract class that provides the infrastructure for processing URLs with a specific URL scheme. You create subclasses for any custom protocols or URL schemes that your app supports.
大概意思是NSURLProtocol是用来处理URL数据加载的类。它本身是一个抽象类,并为处理网络请求提供基础一个壳,你可以子类化它,重新定义新的或已存在的URL的行为。
可能还是不明白有什么鬼用,接下来我们来看一张图,你就能很容易明白他在整个系统中地位了。这张图可能有点老了,没有把NSURLSession 放进来.
虽然URL加载系统包含很多东西,一层又一层的装饰设计,造就庞大的URL Loading System。NSURLProtocol或许是URL加载系统中最功能强大但同时也是最晦涩的部分了。用了它,你不必改动应用在网络调用上的其他部分,就可以改变URL加载行为的全部细节。 对现有的AFNetworking 、SDWebImage 等你都可以重新修改其中网络请求和响应。
- 比如拦截图片加载请求,转而从本地文件加载。之前半糖的文章HTML发布文章预览功能就是通过他实现的
- 对请求进行HTTPDNS解析,动态设置Host,解决不同网络下客户端不能访问的情况
- 对请求和返回的数据进行敏感信息过滤
- 对请求进行验证签名
- 自定义网络请求和返回结果
- 实现网络缓存
1、 创建NSURLProtocol的子类,重写以下方法
+ canInitWithRequest: 抽象方法,子类给出是否能相应该请求。如果响应YES,说明自己的CustomURLProtocol实现该请求。
+ canonicalRequestForRequest:抽象方法,重写该方法,可以对请求进行修改,例如添加新的头部信息,修改,修改url等,返回修改后的请求。
+ requestIsCacheEquivalent:toRequest:Returns whether two requests are equivalent for cache purposes.
- startLoading开始下载,需要在该方法中发起一个请求,对于NSURLConnection来说,就是创建一个NSURLConnection,对于NSURLSession,就是发起一个NSURLSessionTask 。一般下载前需要设置该请求正在进行下载,防止多次下载的情况发生
<pre class=”brush: java; gutter: true;”>
[NSURLProtocol setProperty:@(YES) forKey:protocolKey inRequest:request];
</pre>
- - stopLoading 停止相应请求,清空请求Connection 或 Task
2、定义自己的网络请求
实现startLoading中代理协议方法。 对NSURLConnection来说,要实现NSURLConnectionDataDelegate代理方法,对于NSURLSession来说,需要实现NSURLSessionTaskDelegate代理方法。在这些协议方法中需要通过NSURLProtocolClient 通知URL Loading System
/*-URLProtocol:cachedResponseIsValid:
-URLProtocol:didCancelAuthenticationChallenge:
-URLProtocol:didFailWithError:
-URLProtocol:didLoadData:
-URLProtocol:didReceiveAuthenticationChallenge:
-URLProtocol:didReceiveResponse:cacheStoragePolicy:
-URLProtocol:wasRedirectedToRequest:redirectResponse:
-URLProtocolDidFinishLoading:
*/
3、合适的位置注册自定义的NSURLProtocol子类
调用[NSURLProtocol registerClass:[CustomURLProtocol class]]
进行注册,调用[NSURLProtocol unregisterClass:[CustomURLProtocol class]]注销
结束
- 自定义HttpDNSImageURLProtocol 实现了对SDWebImage 框架中图片加载连接的拦截,通过拦截请求域名动态解析出对应的访问IP,重设请求Host,优化图片加载。
-源码下载地址
参考
- http://nshipster.cn/nsurlprotocol/
- https://developer.apple.com/reference/foundation/nsurlprotocol?language=objc
- http://www.cnblogs.com/fengmin/p/5484413.html
- http://stackoverflow.com/questions/5572258/ios-webview-remote-html-with-local-image-files
- http://www.infinite-loop.dk/blog/2011/09/using-nsurlprotocol-for-injecting-test-data/
- //www.greatytc.com/p/7c89b8c5482a
- https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html#//apple_ref/doc/uid/10000165i