AJAX(XMLHttpRequest)跨域请求笔记(一)

以下代码请在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(跨域资源共享,该规范地址:http://www.w3.org/TR/access-control/和http: //dev.w3.org/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 数据)随同请求一起发送。

html

  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"  
  2.  "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">   
  3. < html   xmlns = "http://www.w3.org/1999/xhtml" >   
  4. < head >   
  5. < title > AJAX跨域请求测试 </ title >   
  6. </ head >   
  7. < body >   
  8. < input   type = 'button'   value = '开始测试'   onclick = 'crossDomainRequest()'   />   
  9. < div   id = "content" > </ div >   
  10. < mce:script   type = "text/javascript" > <!--  
  11.     var xhr  =  new  XMLHttpRequest();  
  12.     var url  =  'http://www.cnpeng.org/cross_site_request.php' ;  
  13.     function crossDomainRequest() {  
  14.       document.getElementById("content").innerHTML  =  "开始……" ;  
  15.       if (xhr) {  
  16.         xhr.open('GET', url, true);  
  17.         xhr.onreadystatechange  =  handler ;  
  18.         xhr.send();  
  19.       } else {  
  20.         document.getElementById("content").innerHTML  =  "不能创建 XMLHttpRequest" ;  
  21.       }  
  22.     }  
  23.     function handler(evtXHR) {  
  24.       if (xhr.readyState  == 4) {  
  25.         if (xhr.status  == 200) {  
  26.           var response  =  xhr .responseText;  
  27.           document.getElementById("content").innerHTML  =  "结果:"  + response;  
  28.         } else {  
  29.           document.getElementById("content").innerHTML  =  "不允许跨域请求。" ;  
  30.         }  
  31.       }  
  32.       else {  
  33.         document.getElementById("content").innerHTML += "< br /> 执行状态 readyState:" + xhr.readyState;  
  34.       }  
  35.     }  
  36. // --> </ mce:script >   
  37. </ body >   
  38. </ html >   

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/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 = 'http://www.cnpeng.org/cross_site_request.php'; function crossDomainRequest() { document.getElementById("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.getElementById("content").innerHTML = "结果:" + response; } else { document.getElementById("content").innerHTML = "不允许跨域请求。"; } } else { document.getElementById("content").innerHTML += "<br/>执行状态 readyState:" + xhr.readyState; } } // --></mce:script> </body> </html>

被访问页面,设置header为“Access-Control-Allow-Origin”

cross_site_request.php

  1. <?php  
  2. header('Content-Type:text/html;charset=GB2312' );  
  3. header("Access-Control-Allow-Origin:*" );  
  4. echo   "你的第一个跨域测试成功啦!" ;  
  5. ?>  
<?php header('Content-Type:text/html;charset=GB2312'); header("Access-Control-Allow-Origin:*"); echo "你的第一个跨域测试成功啦!"; ?>

需要特别注意的是:在请求信息中,浏览器使用 Origin 这个 HTTP 头来标识该请求来自于 http://www.cnpeng.org:8080;在返回的响应信息中,使用 Access-Control-Allow-Origin 头来控制哪些域名的脚本可以访问该资源。如果设置 Access-Control-Allow-Origin:*,则允许所有域名的脚本访问该资源。如果有多个,则只需要使用逗号分隔开即可。

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

;