JS跨域访问解决方案总结
0引言:
跨域请求,顾名思义,就是一个站点中的资源去访问另外一个不同域名站点上的资源。这种情况很常见,比如说通过style标签加载外部样式表文件、通过 img 标签加载外部图片、通过 script 标签加载外部脚本文件、通过 Web font 加载字体文件等等。默认情况下,脚本访问文档属性等数据采用的是同源策略(Same origin policy
同源策略:如果两个页面的协议、域名和端口是完全相同的,那么它们就是同源的。同源策略是为了防止从一个地址加载的文档或脚本访问或者设置从另外一个地址加载的文档的属性。如果两个页面的主域名相同,则还可以通过设置document.domain 属性将它们认为是同源的。
随着 Web2.0 SNS 的兴起,Web 应用对跨域访问的需求也越来越多,但在脚本中进行跨域请求是受安全性限制的,Web 开发人员迫切需要提供一种更安全、方便的跨域请求方式来融合(Mashup)自己的 Web 应用。这样做的一个好处就是可以将请求分摊到不同的服务器,减轻单个服务器压力以提高响应速度;另外一个好处是可以将不同的业务逻辑分布到不同的服务器上以降低负载。
值得庆幸的是,跨域请求的标准已经出台,主流浏览器也已经实现了这一标准。W3C 工作组中的 Web Applications Working GroupWeb 应用工作组)发布了一个 Cross-Origin Resource Sharing(跨域资源共享规范)推荐规范来解决跨域请求的问题。该规范提供了一种更安全的跨域数据交换方法。具体规范的介绍可以访问上面提供的网站地址。值得注意的是:该规范只能应用在类似XMLHttprequest 这样的 API 容器内。IE8Firefox 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,只使用 GETPOST 进行的请求,这里的POST只包括发送给服务器的数据类型Content-Type)必须是 application/x-www-form-urlencodedmultipart/form-data 或者 text/plain中一个。
bHTTP 请求没有设置自定义的请求头,如我们常用的 X-JSON
先使用下面的代码进行测试:
XML/HTML代码
1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"   
2.  "/TR/xhtml1/DTD/xhtml1-transitional.dtd>   
3. <html xmlns="/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 = '/SimpleCrossSiteRequests.aspx';   
13.     function crossDomainRequest() {   
14.       ElementById("content").innerHTML = "开始……";   
15.       if (xhr) {   
16.         xhr.open('GET', url, true);   
17.         adystatechange = handler;   
18.         xhr.send();   
19.       } else {   
20.         ElementById("content").innerHTML = "不能创建 XMLHttpRequest";   
21.       }   
22.     }   
23.     function handler(evtXHR) {   
24.       if (adyState == 4) {   
25.         if (xhr.status == 200) {   
26.           var response = xhr.responseText;   
27.           ElementById("content").innerHTML = "结果:" + response;   
28.         } else {   
29.           ElementById("content").innerHTML = "不允许跨域请求。";   
30.         }   
31.       }   
32.       else {   
33.         ElementById("content").innerHTML += "<br/>执行状态 readyState" +&adyState;   
34.       }   
35.     }   
36. // --></mce:script>   
37. </body>   
38. </html>   
然后,在服务器创建 CrossDomainRequest.aspx 的内容如下:
C# 代码
1. <%@ Page Language="C#" %> 
2. <mce:script runat="server"><!-- 
3.   protected void Page_Load(object sender, EventArgs e) 
4.   { 
5.     Response.AddHeader("Access-Control-Allow-Origin""_xian_hui:801"); 
6.     Response.Write("孟宪会向各位朋友发来贺电:你的第一个跨域测试成功啦!!!"); 
7.   } 
8. // --></mce:script> 
点击 “开始测试 按钮,发送的请求和返回的响应信息如下:
XML/HTML代码
1. GET /SimpleCrossSiteRequests.aspx HTTP/1.1 
2. Host:  
3. 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) 
4. Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8 
5. Accept-Language: zh-cn,zh;q=0.5 
6. Accept-Encoding: gzip,deflate 
7. Accept-Charset: GB2312,utf-8;q=0.7,*;q=0.7 
8. Keep-Alive: 300 
9. Connection: keep-alive 
10. Referer: _xian_hui:801/CrossDomainAjax/SimpleCrossSiteRequests.html 
11. Origin: _xian_hui:801 
12. HTTP/1.x 200 OK 
13. Date: Sun, 10 Jan 2010 13:52:00 GMT 
14. Server: Microsoft-IIS/6.0 
15. X-Powered-By: ASP.NET 
16. X-AspNet-Version: 2.0.50727 
17. Access-Control-Allow-Origin: _xian_hui:801 
18. Set-Cookie: ASP.NET_SessionId=wk5v5nrs5wbfi4rmpjy2jujbpath=/; HttpOnly 
19. Cache-Control: private