2019独角兽企业重金招聘Python工程师标准>>>
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等等。
}