当前位置:文档之家› AJAX(XMLHttpRequest)进行跨域请求方法详解

AJAX(XMLHttpRequest)进行跨域请求方法详解

发表于:2010-1-11 浏览:965 作者:转载遗失来源:网络转载关键字:AJAX,详解,请求描述:注意:以下代码请在Firefox3.5、Chrom e3.0、Safari4之后的版本中进行测试。

IE8的实现方法与其他浏览不同。

跨域请求,顾名思义,就是一个站点中的资源去访问另外一个不同域名站点上的资源。

这种情况很常见注意:以下代码请在Firefox 3.5、Chrome 3.0、Safari 4之后的版本中进行测试。

IE8的实现方法与其他浏览不同。

跨域请求,顾名思义,就是一个站点中的资源去访问另外一个不同域名站点上的资源。

这种情况很常见,比如说通过style 标签加载外部样式表文件、通过img 标签加载外部图片、通过 script 标签加载外部脚本文件、通过Webfont 加载字体文件等等。

默认情况下,脚本访问文档属性等数据采用的是同源策略(Same origin policy)。

那么,什么是同源策略呢?如果两个页面的协议、域名和端口是完全相同的,那么它们就是同源的。

同源策略是为了防止从一个地址加载的文档或脚本访问或者设置从另外一个地址加载的文档的属性。

如果两个页面的主域名相同,则还可以通过设置document.domain 属性将它们认为是同源的。

随着Web2.0 和SNS 的兴起,Web 应用对跨域访问的需求也越来越多,但是,在脚本中进行跨域请求是受安全性限制的,Web 开发人员迫切需要提供一种更安全、方便的跨域请求方式来融合(Mashup)自己的Web 应用。

这样做的一个好处就是可以将请求分摊到不同的服务器,减轻单个服务器压力以提高响应速度;另外一个好处是可以将不同的业务逻辑分布到不同的服务器上以降低负载。

值得庆幸的是,跨域请求的标准已经出台,主流浏览器也已经实现了这一标准。

W3C 工作组中的 Web Applications Working Group(Web 应用工作组)发布了一个Cross-Origin Resource Sharing(跨域资源共享,该规范地址:/TR/access-control/和/2006/waf/access-control/)推荐规范来解决跨域请求的问题。

该规范提供了一种更安全的跨域数据交换方法。

具体规范的介绍可以访问上面提供的网站地址。

值得注意的是:该规范只能应用在类似XMLHttprequest 这样的API 容器内。

IE8、Firefox 3.5 及其以后的版本、Chrome浏览器、Safari 4 等已经实现了Cross-Origin Resource Sharing 规范,已经可以进行跨域请求了。

Cross-Origin Resource Sharing 的工作方式是通过添加HTTP 头的方法来判断哪些资源允许 Web 浏览器访问该域名下的信息。

然而,对于那些HTTP 请求导致用户数据产生副作用的请求方法(特别是对于除了GET、某些MIME 类型的POST 之外的HTTP方法),该规范要求浏览器对请求进行“预先验”,通过发送HTTP 的OPTIONS 请求头询问服务器有哪些支持的方法,在征得服务器的同意后,再使用实际的HTTP 请求方法发送实际的请求。

服务器也可以通知客户端是否需要将验证信息(如Cookie 和HTTP Authentication 数据)随同请求一起发送。

下面我们就采用实际的例子说明Cross-Origin Resource Sharing 是如何工作的。

1,简单请求什么样的请求算是简单请求呢?简单请求必须满足下面2点:a,只使用GET、POST 进行的请求,这里的POST只包括发送给服务器的数据类型(Content-Type)必须是application/x-www-form-urlencoded、multipart/form-data 或者text/plain中一个。

b,HTTP 请求没有设置自定义的请求头,如我们常用的X-JSON。

先使用下面的代码进行测试:<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN""/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="/1999/xhtml"><head><title>孟宪会之AJAX跨域请求测试</title></head><body><input type=…button… value=…开始测试… onclick=…crossDomainRequest()… /><div id="content"></div><mce:script type="text/javascript"><!--var xhr = new XMLHttpRequest();var url = …/SimpleCrossSiteRequests.aspx…;function crossDomainRequest() {document.getElem entById("content").innerHTML = "开始……";if (xhr) {xhr.open(…GET…, url, true);xhr.onreadystatechange = handler;xhr.send();} else {document.getElementById("content").innerHTML = "不能创建XMLHttpRequest";}}function handler(evtXHR) {if (xhr.readyState == 4) {if (xhr.status == 200) {var response = xhr.responseText;document.getElem entById("content").innerHTML = "结果:" + response;} else {document.getElem entById("content").innerHTML = "不允许跨域请求。

";}}else {document.getElementById("content").innerHTML += "<br/>执行状态readyState:" + xhr.readyState;}}// --></mce:script></body></html>然后,在服务器创建CrossDomainRequest.aspx 的内容如下:<%@ Page Language="C#" %><mce:script runat="server"><!--protected void Page_Load(object sender, EventArgs e){Response.AddHeader("Access-Control-Allow-Origin","http://www.meng_xian_:801");Response.Write("孟宪会向各位朋友发来贺电:你的第一个跨域测试成功啦!!!");}// --></mce:script>点击“开始测试” 按钮,发送的请求和返回的响应信息如下:GET /SimpleCrossSiteRequests.aspx HTTP/1.1Host: User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.2; zh-CN; rv:1.9.1.7) Gecko/20091221 Firefox/3.5.7 (.NET CLR 3.5.30729)Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8Accept-Language: zh-cn,zh;q=0.5Accept-Encoding: gzip,deflateAccept-Charset: GB2312,utf-8;q=0.7,*;q=0.7Keep-Alive: 300Connection: keep-aliveReferer:http://www.meng_xian_:801/CrossDomainAjax/SimpleCrossSiteRequests.html Origin: http://www.meng_xian_:801HTTP/1.x 200 OKDate: Sun, 10 Jan 2010 13:52:00 GMTServer: Microsoft-IIS/6.0X-Powered-By: X-AspNet-Version: 2.0.50727Access-Control-Allow-Origin: http://www.meng_xian_:801Set-Cookie: _SessionId=wk5v5nrs5wbfi4rmpjy2jujb; path=/; HttpOnlyCache-Control: privateContent-Type: text/html; charset=utf-8Content-Length: 84需要特别注意的是:在请求信息中,浏览器使用Origin 这个HTTP 头来标识该请求来自于http://www.meng_xian_:801;在返回的响应信息中,使用Access-Control-Allow-Origin头来控制哪些域名的脚本可以访问该资源。

如果设置Access-Control-Allow-Origin:*,则允许所有域名的脚本访问该资源。

如果有多个,则只需要使用逗号分隔开即可。

注意:在服务器端,Access-Control-Allow-Origin 响应头 http://www.meng_xian_:801 中的端口信息不能省略。

有人可能会想:自己发送请求头会如何呢?比如xhr.setRequestHeader("Origin","http://www.meng_xian_:801"); 实践证明,自己设置Origin 头是不行的。

相关主题