本文转载至 http://blog.csdn.net/u014011807/article/details/39894247

NSURLProtocol 是iOS中非常重要的一个部分,我们经常会在以下地方用到它:

(1)网络请求代理转发(FQ 网络加速等)

(2)离线缓存策略

总之,只要是需要对本地的URL请求进行截获的,我们都需要使用这个东东。

IOS中我们经常使用的网络请求NSURLConnection以及WebView的页面加载都会被NSURLProtocol截获,因此这个部分的核心就是如何来用这个部分:

下面我将根据使用的步骤来描述NSURLProtocol的使用过程:

最终运行结果:

(1)首先我们需要创建一个类,继承NSURLProtocol

并且实现代理继承:

@interfaceMyURLProtocol () <NSURLConnectionDelegate>

@end

(2)注册这个协议

一般情况下,这个协议在AppDelegate文件中注册:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions

{

[NSURLProtocol  registerClass:[MyURLProtocolclass]];

// Override point for customization after application launch.

returnYES;

}

下面就可以使用这个东东了

(3)下面准备好一段WebView的页面加载代码,这个太简单了,不详细说了,给一个参考代码:

- (void) sendRequest {

NSString *text =self.textField.text;

if (![textisEqualToString:@""]) {

NSURL *url = [NSURLURLWithString:text];

NSURLRequest *request = [NSURLRequestrequestWithURL:url];

[self.webView loadRequest:request];

}

}

(4)可以正常的使用这个NSURLProtocol只需要简单的使用下面9个方法,这九个方法是这个协议可以实现的最基本和最重要的几个方法,当然它还有很多高级的功能,这里就不一一介绍了。

最重要的9个方法如下:

<1>

@method:创建NSURLProtocol实例,NSURLProtocol注册之后,所有的NSURLConnection都会通过这个方法检查是否持有该Http请求。

@parma :

@return: YES:持有该Http请求NO:不持有该Http请求

+ (BOOL)canInitWithRequest:(NSURLRequest *)request

#pragma mark --NSURLProtocol Hold RelevantMethod 4个方法

<2-5>

@method: NSURLProtocol抽象类必须要实现。通常情况下这里有一个最低的标准:即输入输出请求满足最基本的协议规范一致。因此这里简单的做法可以直接返回。一般情况下我们是不会去更改这个请求的。如果你想更改,比如给这个request添加一个title,组合成一个新的http请求。

@parma :本地HttpRequest请求:request

@return:直接转发

+ (NSURLRequest*)canonicalRequestForRequest:(NSURLRequest *)request

@method: NSURLProtocol缓存系统设置:如果有两个URL请求,并且他们是相等的,那么这里可以使用相同的缓存空间

@parma :本地HttpRequest请求:request

@return:

+ (BOOL)requestIsCacheEquivalent:(NSURLRequest *)a toRequest:(NSURLRequest*)b

@method:获取网站上的数据建立connect连接

@parma :

@return:

- (void)startLoading

@method:当前Connection连接取消的时候被调用。尤其要注意这个StopLoading方法,在本地NSURLRequest初始化的时候,有一个超时时间,在低速网络下,有可能页面还没来得及加载完,这个StopLoading方法就被调用了。

@parma :

@return:

- (void)stopLoading

<6-9>接收数据

#pragma mark --NSURLProtocol Delegate 4个方法

-                 (void)connection:(NSURLConnection*)connection didReceiveResponse:(NSURLResponse *)response

-                 (void)connection:(NSURLConnection*)connection didReceiveData:(NSData*)data

-                 (void)connectionDidFinishLoading:(NSURLConnection*)connection

-                 (void)connection:(NSURLConnection*)connection didFailWithError:(NSError*)error

注意这里的方法中:

canInitWithRequest 会创建一个实例,YES时候,然后继续调用startLoading。这里会继续调用canInitWithRequest。陷入死循环,因此这里我们常用的做法是设置一个[NSURLProtocolsetProperty:@YESforKey:@"MyURLProtocolHandledKey"inRequest:newRequest];

这样就可以避免程序陷入死循环。

(5)下面给出一段参考代码:

@implementation MyURLProtocol

/**

@method: 创建NSURLProtocol实例,NSURLProtocol注册之后,所有的NSURLConnection都会通过这个方法检查是否持有该Http请求。

@parma :

@return: YES:持有该Http请求 NO:不持有该Http请求

*/

+ (BOOL)canInitWithRequest:(NSURLRequest *)request {

staticNSUInteger requestCount =0;

NSLog(@"Request #%u: URL = %@", requestCount++, request);

if([NSURLProtocolpropertyForKey:@"MyURLProtocolHandledKey"inRequest:request]) {

returnNO;

}

returnYES;

}

#pragma mark --NSURLProtocol Hold Relevant Method

/**

@method: NSURLProtocol抽象类必须要实现。通常情况下这里有一个最低的标准:即输入输出请求满足最基本的协议规范一致。因此这里简单的做法可以直接返回。一般情况下我们是不会去更改这个请求的。如果你想更改,比如给这个request添加一个title,组合成一个新的http请求。

@parma : 本地HttpRequest请求:request

@return: 直接转发

*/

+ (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest*)request {

return request;

}

/**

@method: NSURLProtocol缓存系统设置:如果有两个URL请求,并且他们是相等的,那么这里可以使用相同的缓存空间

@parma : 本地HttpRequest请求:request

@return:

*/

+ (BOOL)requestIsCacheEquivalent:(NSURLRequest *)a toRequest:(NSURLRequest *)b {

return [superrequestIsCacheEquivalent:atoRequest:b];

}

- (void)startLoading {

NSMutableURLRequest *newRequest = [self.requestmutableCopy];

[NSURLProtocolsetProperty:@YESforKey:@"MyURLProtocolHandledKey"inRequest:newRequest];

self.connection = [NSURLConnectionconnectionWithRequest:newRequestdelegate:self];

}

/**

@method: 当前Connection连接取消的时候被调用

@parma :

@return:

*/

- (void)stopLoading {

[self.connectioncancel];

self.connection =nil;

}

#pragma mark --NSURLProtocol Delegate

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {

[self.clientURLProtocol:self didReceiveResponse:response cacheStoragePolicy:NSURLCacheStorageNotAllowed];

}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {

[self.clientURLProtocol:selfdidLoadData:data];

}

- (void)connectionDidFinishLoading:(NSURLConnection *)connection {

[self.clientURLProtocolDidFinishLoading:self];

}

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {

[self.clientURLProtocol:selfdidFailWithError:error];

}

@end

最新文章

  1. javascript (2)
  2. error C2678
  3. js作用域和变量提升
  4. NS 802.11函数分析(一)
  5. jquery过滤器之:contains()、.filter()
  6. Grunt完成对LESS实时编译
  7. 降龙十八掌之一:(亢龙有悔)SQL Server Profiler和数据库引擎优化顾问
  8. No Dialect mapping for JDBC type: -9
  9. DVDRW光驱无法读DVD刻录盘
  10. Elasticsearch升级至1.x后API的变化-三
  11. Spark 初级算子
  12. Eval()、XPath() 和 Bind() 这类数据绑定方法只能在数据绑定控件的上下文中使用
  13. python demo整理
  14. UWP Composition API - New FlexGrid 锁定行列
  15. CrawlScript脚本语言实现网络爬虫
  16. KBEngine WebConsole Guide
  17. Linux CentOS 安装MySql以及搭建MySql主从复制
  18. Thymeleaf中的&amp;&amp;解析问题
  19. Django--Auth 模块
  20. 11g新特性-自动sql调优(Automatic SQL Tuning)

热门文章

  1. Sanjeev Arora
  2. 【吉比特】G-bits2018校园春季招聘技术类岗位笔试经验
  3. Swing用于开发Java应用程序用户界面
  4. SpringMVC自定义视图Excel视图和PDF视图
  5. Yii2 中cookie的用法(1)
  6. 10.11 Linux网络相关 10.12 firewalld和netfilter 10.13 netfilter5表5链介绍 10.14 iptables语法
  7. dropwizard metrics - 基本使用介绍
  8. 周期性调度器scheduler_tick
  9. MongoDB:通过mongodump【时间一致性】备份,快速创建secondary复制集节点——更精简的方式2
  10. CorelDRAW中如何复制对象属性详解