本文转载至 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

IOS网络篇1之截取本地URL请求(NSURLProtocol)的更多相关文章

  1. iOS网络篇

    iOS网络请求三步: 1.新建URL连接 2.新建请求(请求新建的URL连接) 3.建立连接. 然后就可以获取数据了. 一.同步GET请求方法 -(void)synchronizationGet { ...

  2. ios -网络加载json和本地加载json

    1网络加载json的时候,要在模型的实现文件里写: - (void)setValue:(id)value forKey:(NSString *)key { } 2本地加载json的时候,要在模型的实现 ...

  3. IOS网络第二天 - 01-基本的HTTP请求

    ***************** #import "HMViewController.h" #import "MBProgressHUD+MJ.h" @int ...

  4. IOS网络第一天-01基本的HTTP请求

    *************************** #import "HMViewController.h" #import "MBProgressHUD+MJ.h& ...

  5. iOS网络相关零散知识总结

    iOS网络相关零散知识总结 1. URL和HTTP知识 (1) URL的全称是Uniform Resource Locator(统一资源定位符). URL的基本格式 = 协议://主机地址/路径   ...

  6. iOS开发网络篇—搭建本地服务器

    iOS开发网络篇—搭建本地服务器 一.简单说明 说明:提前下载好相关软件,且安装目录最好安装在全英文路径下.如果路径有中文名,那么可能会出现一些莫名其妙的问题. 提示:提前准备好的软件 apache- ...

  7. iOS开发网络篇—HTTP协议

    iOS开发网络篇—HTTP协议 说明:apache tomcat服务器必须占用8080端口 一.URL 1.基本介绍 URL的全称是Uniform Resource Locator(统一资源定位符) ...

  8. iOS开发网络篇—文件的上传

    iOS开发网络篇—文件的上传 说明:文件上传使用的时POST请求,通常把要上传的数据保存在请求体中.本文介绍如何不借助第三方框架实现iOS开发中得文件上传. 由于过程较为复杂,因此本文只贴出部分关键代 ...

  9. 02.iOS开发网络篇—HTTP协议

    iOS开发网络篇—HTTP协议 说明:apache tomcat服务器必须占用8080端口 一.URL 1.基本介绍 URL的全称是Uniform Resource Locator(统一资源定位符) ...

随机推荐

  1. 关于DLNA

    Version:0.9 StartHTML:-1 EndHTML:-1 StartFragment:00000099 EndFragment:00005587  概念 DLNA的全称是DIGITAL ...

  2. Linux操作_磁盘管理_增加虚拟磁盘

    环境:虚拟机 VM 12,Linux版本号 CentOS 7.3 1,在当前的虚拟机选项卡点击鼠标右键,选择“设置” 2,在弹出的对话框中左侧选中“磁盘”->点击下方“添加”按钮,在弹出的“添加 ...

  3. C#基础---------------C#正则表达式2

    C#正则表达式语法规则详解 正则表达式基础知识        一个正则表达式就是由普通字符(例如字符 a 到 z)以及特殊字符(称为元字符)组成的文字模式.   字母文本指的是普通文本如"a ...

  4. ubuntu -- 系统目录结构

    1./:目录属于根目录,是所有目录的绝对路径的起始点,Ubuntu 中的所有文件和目录都在跟目录下. 2./etc:此目录非常重要,绝大多数系统和相关服务的配置文件都保存在这里,这个目录的内容一般只能 ...

  5. Android Notification和权限机制探讨

    近期为了在部门内做一次小型的技术分享.深入了解了一下Notification的实现原理.以及android的权限机制.在此做个记录.文章可能比較长,没耐心的话就直接看题纲吧. 先看一下以下两张图 图一 ...

  6. 洞悉linux下的Netfilter&iptables:什么是Netfilter?

    本人研究linux的防火墙系统也有一段时间了,由于近来涉及到的工作比较纷杂,久而久之怕生熟了.趁有时间,好好把这方面的东西总结一番.一来是给自己做个沉淀,二来也欢迎这方面比较牛的前辈给小弟予以指点,共 ...

  7. fedora26在编译s3c2440内核时make menuconfig *** Unable to find the ncurses libraries

    [root@fedora-26 linux-2.6.32.2]# make menuconfig *** Unable to find the ncurses libraries or the *** ...

  8. YFCMF 问题

    1.菜单不见了,yf.php  (main 改为0 ) function tagMenu $parseStr .='echo get_menu("main","'.$to ...

  9. 性能监控-TP理解

    首先给出Google到的答案: The tp90 is a minimum time under which 90% of requests have been served. tp90 = top ...

  10. ABBYY FineReader 12中的用户模式你会用吗

    在ABBYY FineReader 12OCR文字识别软件中,有一个概念叫“训练”,它是在字符图像和字符本身之间建立对应关系的过程,训练模式可以提高含有装饰字体的文档或包含特殊字符(例如数学符号)文档 ...