与HTTP服务器通信(Communicating with HTTP Servers)

本文解释了如何创建、发送和接收HTTP请求和响应。

创建一个CFHTTP请求

HTTP请求是一个消息,这个消息由远程服务器执行的方法,操作的对象(URL),消息头和消息体。方法通常是下面之一:GET, HEAD, PUT, POST, DELETE, TRACE, CONNECT 或OPTIONS。用CFHTTP创建一个HTTP请求分为四个步骤:

  • 使用CFHTTPMessageCreateRequest 函数生成CFHTTP消息对象
  • 使用CFHTTPMessageSetBody函数设置消息体
  • 使用CFHTTPMessageSetHeaderFieldValue 函数设置消息头
  • 通过调用CFHTTPMessageCopySerializedMessage函数序列化消息

示例代码类似清单3-1中的代码。

清单3-1 创建一个HTTP请求

CFStringRef bodyString = CFSTR(""); // Usually used for POST data
CFDataRef bodyData = CFStringCreateExternalRepresentation(kCFAllocatorDefault,
                                        bodyString, kCFStringEncodingUTF8, 0);

CFStringRef headerFieldName = CFSTR("X-My-Favorite-Field");
CFStringRef headerFieldValue = CFSTR("Dreams");

CFStringRef url = CFSTR("http://www.apple.com");
CFURLRef myURL = CFURLCreateWithString(kCFAllocatorDefault, url, NULL);

CFStringRef requestMethod = CFSTR("GET");
CFHTTPMessageRef myRequest =
    CFHTTPMessageCreateRequest(kCFAllocatorDefault, requestMethod, myURL,
                               kCFHTTPVersion1_1);

CFDataRef bodyDataExt = CFStringCreateExternalRepresentation(kCFAllocatorDefault, bodyData, kCFStringEncodingUTF8, 0);
CFHTTPMessageSetBody(myRequest, bodyDataExt);
CFHTTPMessageSetHeaderFieldValue(myRequest, headerFieldName, headerFieldValue);
CFDataRef mySerializedRequest = CFHTTPMessageCopySerializedMessage(myRequest);

CFHTTPMessageCreateRequest 返回的消息对象引用(myRequest)和消息体(bodyData)一起发送到CFHTTPMessageSetBody 。然后调用CFHTTPMessageSetHeaderFieldValue 使用相同消息对象引用,头(headerField)名称和设置的值(value)。头参数是个CFString对象,例如Content-Length,值参数是一个CFString对象例如1260.最后,调用CFHTTPMessageCopySerializedMessage 序列化消息,通过写入流发送到接受者,例子详见http://www.apple.com。在此示例代码中,通过调用CFURLCreateWithString,url 是首先转换成一个CFURL对象。然后调用CFHTTPMessageCreateRequest ,有四个参数:kCFAllocatorDefault 指定默认系统内存分配器用来创建消息应用,requestMethod 指定方法,例如POST方法,myURL 用来指定URL,例如,http://www.apple.com,kCFHTTPVersion1_1指定消息HTTP版本是1.1.

注意:请求主体通常省略。请求主体通常用于一个包含POST数据的POST请求。它也可以用于其他有关HTTP扩展的请求类型,例如WebDAV。更多信息参见RFC 2616

当不再需要消息,释放消息对象并序列化消息。见清单3-2的示例代码

清单3-2 释放一个HTTP请求

CFRelease(myRequest);
CFRelease(myURL);
CFRelease(url);
CFRelease(mySerializedRequest);
myRequest = NULL;
mySerializedRequest = NULL;

创建一个CFHTTP响应

创建一个HTTP响应的步骤与创建一个HTTP请求的步骤几乎完全相同。唯一的区别是,调用函数CFHTTPMessageCreateResponse ,而不是CFHTTPMessageCreateRequest,两者使用相同的参数

反序列化传入的HTTP请求

反序列化一个传入的HTTP请求,使用CFHTTPMessageCreateEmpty 函数,创建一个空消息,isRequest 参数设为TRUE 指定创建一个空的请求消息。然后调用CFHTTPMessageAppendBytes函数将传入的消息添加到空消息中。CFHTTPMessageAppendBytes反序列化消息并移除任何可能包含的控制信息。

继续这样做直到CFHTTPMessageIsHeaderComplete 函数返回TRUE。如果你不检查CFHTTPMessageIsHeaderComplete 是否返回TRUE,消息可能是不完整的或不可信的。清单3-3可以看到这两个函数使用的例子。

清单3-3 反序列化消息

CFHTTPMessageRef myMessage = CFHTTPMessageCreateEmpty(kCFAllocatorDefault, TRUE);
if (!CFHTTPMessageAppendBytes(myMessage, &data, numBytes)) {
    //Handle parsing error
}

在示例中,data 是添加的数据而numBytes 是的data 长度。调用CFHTTPMessageIsHeaderComplete 验证附加消息的头是完整的。

if (CFHTTPMessageIsHeaderComplete(myMessage)) {
    // Perform processing.
}

消息反序列化后,你可以调用如下函数从消息中提取信息:

  • CFHTTPMessageCopyBody 用来获取消息主体
  • CFHTTPMessageCopyHeaderFieldValue 用来获取特定头字段值
  • CFHTTPMessageCopyAllHeaderFields 用来获取所有消息头字段
  • CFHTTPMessageCopyRequestURL 用来获取消息URL
  • CFHTTPMessageCopyRequestMethod 用来获取消息请求方法

当你不再需要该消息,释放并恰当的处理它。

反序列化传入的HTTP响应

正如创建HTTP请求类似于创建HTTP响应一样,反序列化传入的HTTP请求与反序列化传入的HTTP响应类似。唯一重要的区别是,当调用CFHTTPMessageCreateEmpty,你必须给isRequest 参数传入FALSE 来指定将要创建的消息是响应消息。

使用读取流序列化并发送HTTP请求

你可以使用CFReadStream对象来序列化和发送CFHTTP请求。当你使用CFReadStream对象来发送一个CFHTTP请求时,打开流因为消息必须在同一步中序列化好发送。使用CFReadStream 对象来发送CFHTTP请求,使获取请求的响应更加容易,因为响应作为流的属性是可用的。

序列化并发送一个HTTP请求

使用CFReadStream 对象序列化并发送HTTP请求,首先创建一个CFHTTP请求并设置消息主体和头,在创建CFHTTP请求(Creating a CFHTTP Request)中有描述。然后,调用CFReadStreamCreateForHTTPRequest 函数创建一个CFReadStream 对象,传递你刚刚创建的请求。最后,通过CFReadStreamOpen打开读取流。

当调用CFReadStreamCreateForHTTPRequest ,复制一份传入的CFHTTP请求对象。因此,如果有必要,你可以在调用CFReadStreamCreateForHTTPRequest 之后立即释放CFHTTP请求对象。

因为当创建CFHTTP请求时,读取流打开一个套接字连接myUrl 参数指定的服务器,两者之间允许有些时间差。打开读取流也会导致请求被序列化和发送。

清单3-4是一个关于如何序列化和发送HTTP请求的例子

清单3-4 读取流序列化HTTP请求

CFStringRef url = CFSTR("http://www.apple.com");
CFURLRef myURL = CFURLCreateWithString(kCFAllocatorDefault, url, NULL);
CFStringRef requestMethod = CFSTR("GET");

CFHTTPMessageRef myRequest = CFHTTPMessageCreateRequest(kCFAllocatorDefault,
        requestMethod, myUrl, kCFHTTPVersion1_1);
CFHTTPMessageSetBody(myRequest, bodyData);
CFHTTPMessageSetHeaderFieldValue(myRequest, headerField, value);

CFReadStreamRef myReadStream = CFReadStreamCreateForHTTPRequest(kCFAllocatorDefault, myRequest);

CFReadStreamOpen(myReadStream);

检查响应

在你安排请求到运行循环后,你将最终会得到一个头完成回调。在这一点上,你可以调用CFReadStreamCopyProperty 从读取流获取消息响应。

CFHTTPMessageRef myResponse = (CFHTTPMessageRef)CFReadStreamCopyProperty(myReadStream, kCFStreamPropertyHTTPResponseHeader);

你可以通过调用CFHTTPMessageCopyResponseStatusLine函数,从消息响应获取完整状态行:

CFStringRef myStatusLine = CFHTTPMessageCopyResponseStatusLine(myResponse);

通过调用CFHTTPMessageGetResponseStatusCode函数获取消息响应的状态码:

UInt32 myErrCode = CFHTTPMessageGetResponseStatusCode(myResponse);

注意:如果你正在同步使用这个类(没有安排到运行循环),你必须在调用CFReadStreamCopyProperty之前调用CFReadStreamRead ,先读取消息。CFReadStreamRead 调用一直阻塞直到数据可用(或连接失败)。不要在你的主应用线程中使用。

处理身份验证错误

如果CFHTTPMessageGetResponseStatusCode 函数返回的状态码是401(远程服务器需要身份验证信息)或407(代理服务器需要身份验证),你需要将身份验证信息附加到请求并重新发送。关于如何处理身份验证的信息,请查看与需要身份验证HTTP服务器通信( Communicating with Authenticating HTTP Servers)。

处理重定向错误

当CFReadStreamCreateForHTTPRequest 创建一个读取流,默认情况下流的自动重定向是禁用的。如果请求发送的统一资源定位符或URL重定向到另一个URL,发送该请求将导致一个错误,状态码在300到307之间。如果你接收到一个重定向错误,你需要关闭流,创建流,启用重定向并打开流。参见清单3-5.

清单3-5 重定向HTTP流

CFReadStreamClose(myReadStream);
CFReadStreamRef myReadStream =
    CFReadStreamCreateForHTTPRequest(kCFAllocatorDefault, myRequest);
if (CFReadStreamSetProperty(myReadStream, kCFStreamPropertyHTTPShouldAutoredirect, kCFBooleanTrue) == false) {
    // something went wrong, exit
}
CFReadStreamOpen(myReadStream);

当你创建一个读取流,你可能想要启用自动重定向。

取消一个待定请求

一旦请求已经发送,不能阻止远程服务器处理它。然而,你不再关心响应数据,你可以关闭流。

重要:如果另一个线程正在等待某个线程的流的内容,则不要关闭该流。如果你需要终止请求时,你应该使用非阻塞 I/O,在使用流时防止阻塞(Preventing Blocking When Working with Streams)中有描述。确保在关闭流之前从你的运行循环上移除流。

官方原文地址:

https://developer.apple.com/library/ios/documentation/Networking/Conceptual/CFNetwork/CFHTTPTasks/CFHTTPTasks.html#//apple_ref/doc/uid/TP30001132-CH5-SW2

时间: 2024-10-24 17:00:15

与HTTP服务器通信(Communicating with HTTP Servers)的相关文章

ios Communicating with HTTP Servers 与HTTP服务器通信 官方文档翻译(五)

原文地址:developer 时间过的很快,一眨眼周末就过了.人的一生到底怎么样才能过的有意义呢?我一直在找答案.做自己喜欢的事情,爱自己喜欢的人,或是成就一番伟业.都是不错的选择,找到自己想要做的事情,不要放弃寻找.最终生命带走的只能是珍贵的回忆,晚上快乐. 与HTTP服务器通信 本章解释了如何创建.发送和接收HTTP请求和响应. 创建一个CFHTTP请求 组成一个HTTP请求消息的远程服务器执行的方法,对象操作(URL),消息头和消息体.方法通常是下列之一:GET, HEAD, PUT, P

与身份验证HTTP服务器通信(Communicating with Authenticating HTTP Servers)

本文描述了如何利用CFHTTPAuthentication API与需要身份验证的HTTP服务器通信.它解释了如何找到匹配的验证对象和证书,并将它们应用到HTTP请求,然后存储以供以后使用. 一般来说,如果一个HTTP服务器返回一个401或407响应你的HTTP请求,这表明服务器进行身份验证需要证书.在CFHTTPAuthentication API中,每个证书组存储在CFHTTPAuthentication 对象中.因此,每个不同的身份认证服务器和每个不同用户连接的服务器需要一个单独的CFHT

iOS Communicating with Authenticating HTTP Servers 与HTTP服务器通信认证 官方文档翻译(六)

原文地址:developer 召集翻译人手:请加入486111671  ios 官方文档翻译                                                            与HTTP服务器通信认证 本章描述了如何与HTTP服务器身份验证利用CFHTTPAuthentication API.它解释了如何找到匹配验证对象和凭证,将它们应用到一个HTTP请求,并将它们存储供以后使用. 一般来说,如果一个HTTP服务器将返回一个401或407响应后你的HTTP请求

利用webSocket使网页和服务器通信

WebSocket protocol 是HTML5一种新的协议.它实现了浏览器与服务器全双工通信(full-duplex).具体说明请查阅相关资料,下面我来演示一种简单的页面与服务器通信的简单样例. 新建个web工程(基于tomcat-7版本(6以下的版本未实现webSocket功能)) 引入tomcat/lib目录下的tomcat7-websocket.jar和websocket-api.jar添加到classpath中 新建WebSocketConfig.java如下 本次采用注解方式 im

使用Ajax与服务器通信,达到异步通信效果

AJAX = 异步 JavaScript 和 XML,AJAX 是一种用于创建快速动态网页的技术.通过在后台与服务器进行少量数据交换,AJAX 可以使网页实现异步更新.这意味着可以在不重新加载整个网页的情况下,对网页的某部分进行更新.传统的网页(不使用 AJAX)如果需要更新内容,必需重载整个网页面. XMLHttpRequest 对象 所有现代浏览器均支持 XMLHttpRequest 对象(IE5 和 IE6 使用 ActiveXObject),XMLHttpRequest 用于在后台与服务

AFNetworking+Nginx+HTTPS服务器通信

摘要 1.介绍iOS平台用AFNetworking与HTTPS后台接口进行安全通信. 2.介绍后台自签名证书制作步骤. 3.以及Linux平台Nginx配置HTTPS协议接口的Web站点. iOS AFNetworking HTTPS Openssl Nginx 这个关于AFNetworking的HTTPS安全通信的问题,很多没有过第一次经验的以及甚至有过一次经验的都会有点不确定. 其实很简单: A.对于后台服务器所配置动证书如果是经过CA机构认证颁发的,那么用户用AFNetworking来访问

BS网站在线升级(服务器通信)

背景:日前公司需要将客户企业站增加在线升级功能.即客户登录自身网站管理后台后台,发生请求到我公司门户网站,如果存在新版本则提示用户更新.客户从我们公司买到的空间.数据库.和网站代码后可以直接根据智能提示完成webconfig配置,后期客户可以从我们手中购买网站模版实现个性化风格主题的网站设置.我们所有客户的网站结构是一模一样的.因为前台客户浏览的页面均为代码自动生成的静态页,所以更新过程不影响访客浏览. 分析:BS架构项目不同与CS架构——通过请求检测版本更新后返回更新包,根据本地安装目录即可完

客户端(android,ios)与服务器通信

android,ios客户端与服务器通信为了便于理解,直接用PHP作为服务器端语言 其实就是一个 http请求响应的过程序,先从 B/S模式说起浏览器发起http请求,服务器响应请求,并把数据返回给浏览器,经过浏览器的渲染,即为我们所看到的效果 客户端(android,ios)android通过 httpClient或urlconnection等网络请求去实现http请求,(相当于浏览器,发送 url请求),即访问服务器,服务器响应请求,并把结果返回给客户端,经过客户端渲染,即为我们看到的效果.

开发交通运输部·车载导航系统中的第一关——硬件终端如何与服务器通信——玩转通信协议(源码下载)

一.引子与协议说明 之前开发了一个项目——车载导航系统.遇到的第一个问题就是硬件设备如何与服务器通信. 关键在于通信协议! 众所周知:要想实现通信,首先通信双方就要达成通信协议. 话不多说,且看协议:  ————————————————华丽的分割线————————————————— 以上的这些协议说明是不是看得很头大呢? 遵循如此这般的通信协议的硬件设备又如何才能与服务器以及PC顺利通信呢? 还请各位看官稍安勿躁!且听我娓娓道来! 二.基础知识-TCP与粘包 我们都知道,互联网的核心是TCP/I