1.前言

     ASIHttprequest 是基于CFNetwork的,由于CFNetwork是比较底层的http库,功能比较少,因此,在ASIHttprequest中实现了http协议中比较多的功能,包括代理、gzip、认证、缓存等等。目前,虽然ASIHTTPRequest已经不如前两年那么流行,但是分析一下其代码,对掌握CFNetwork库和HTTP协议还是有好处的,本文将简单分析一下ASIHTTPRequest中几个主要函数的流程。
 

2.处理HTTP Request的主要函数

ASIHTTPRequest::main流程如下(只列举了主要工作):

{
     (1)若允许后台运行,则调用beginBackgroundTaskWithExpirationHandler允许程序后台运行十分钟
     (2) 调用buildPostBody函数构造post的body部分,该函数有两个,一个是基类中的,主要负责压缩body数据,另一个是派生类ASIFormDataRequest中,分别针对post表单和post文件的方式,分别设置content-type为application/x-www-form-urlencoded和multipart/form-data。
     (3)根据url和请求的方法来创建CFHTTPMessageRef对象。
     (4)调用buildRequestHeaders函数来构造header部分,这里只是简单的将各个header字段放到一个NSDictionary变量中。
     (5)如果设置了缓存,并且允许从缓存取数据,则从缓存中读取数据,然后返回。
     (6)调用applyAuthorizationHeader向header字典中添加HTTP认证相关的字段。
     (7)调用CFHTTPMessageSetHeaderFieldValue来将header数据添加到CFHTTPMessageRef对象中。
     (8)调用configureProxies来配置代理。
     (9)调用startRequest来发送请求。
}
 

ASIHTTPRequest::startRequest

{

(1)向主线程发送requestStarted消息

(2)如果存在body,且需要post本地文件,则将本地文件读到postBodyReadStream对象中。然后调用CFReadStreamCreateForStreamedHTTPRequest函数,传入之前创建的CFHTTPMessageRef对象和postBodyReadStream对象,来创建一个用来读取response 的CFReadStream对象。

如果是post数据,则先根据shouldCompressRequestBody的值来判断是否要压缩,然后根据postBody的数据来创建一个NSInputStream对象,并赋给postBodyReadStream对象,然后调用CFReadStreamCreateForStreamedHTTPRequest,传入之前的header和stream对象,来创建NSReadStream对象。

如果不存在body,则直接通过CFReadStreamCreateForHTTPRequest函数来创建NSReadStream对象。

(3)针对https的情况,调用CFReadStreamSetProperty进行设置

(4)如果请求中设置了代理,则调用CFReadStreamSetProperty对stream进行代理相关的设置

(5)处理http持久连接相关的设置

(6)调用scheduleInRunLoop,将readStream对象放入runloop中

(7)调用CFReadStreamSetClient函数来将readStream关联到一个回调函数ReadStreamClientCallBack中,并使用CFReadStreamOpen打开readStream对象

(8)调用进度通知相关的函数

(9)创建一个计时器,用来调用updateStatus函数来更新进度,并将计时器放入当前runloop。

}

3.处理HTTP Response的主要函数

ASIHTTPRequest::handleNetworkEvent (该函数用来处理回调事件)

{

(1)当收到kCFStreamEventHasBytesAvailable 事件时 ,调用handleBytesAvailable(此时表示下层已经读到了response里的数据,这数据可能包含全部的header也可能header尚未读完)

(2)kCFStreamEventEndEncountered,调用handleStreamComplete,此时表示全部的数据包括header和body都已经读完,而且对应chunked数据,底层也已经将其合并完。

(3)kCFStreamEventErrorOccurred事件,调用handleStreamError处理错误

}

ASIHTTPRequest::handleBytesAvailable

{

(1)如果responseHeader对象尚未赋值,则调用readResponseHeaders读取header

(2)申请一块buffer,读取readStream对象里面的数据,此时如果能读到数据,则表示header已经读完了,当前读到的是body里面数据(因为header不是用read方法读的),如果读不到数据,则表示还没有收到body,则返回。

(3)读取到数据之后,如果header里面显示数据是压缩过的,则进行解压缩

(4)解压出数据之后有三种处理方式:

如果用户设置了didReceiveDataSelector或者dataReceivedBlock,这就表示用户希望自己处理每次得到的data,则向主线程发送passOnReceivedData消息。

如果用户在request中设置了下载路径,则将数据写到文件中

如果以上都不满足,则将数据append到rawResponseData中。

}

ASIHTTPRequest::readResponseHeaders

{

(1)使用CFReadStreamCopyProperty从readStream对象中读取header,创建一个CFHTTPMessageRef对象,并且使用CFHTTPMessageIsHeaderComplete检查该对象,判断header是否已经读完,若没有读完,则销毁该对象并返回

(2) 使用CFHTTPMessageCopyAllHeaderFields从CFHTTPMessageRef读出header到一个dictionary中

(3)如果有缓存,且允许读取缓存,则从缓存中读取header并返回。

(4)根据header中的状态码来判断是否需要进行http认证,如果需要则处理认证相关的工作

(5) 从header中content-type,用于对body进行解码,如没有该字段,则使用默认的解码方式对content进行解码

(6)处理cookie相关的工作

(7)如果不需要重定向,则从header中读取content-length,然后根据length做相关处理

(8)处理keepalive相关的工作

(9)最后向主线程发送requestReceivedResponseHeaders通知

}

ASIHTTPRequest::handleStreamComplete

{

该函数做的事情比较简单,主要就是设置各种下载结束的标志、设置读取到的文件大小并发送通知消息、移动下载的临时文件、保存cache等等。

}

 
05-26 12:40