`
zl198751
  • 浏览: 273478 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

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 数据)随同请求一起发送。

下面我们就采用实际的例子说明 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。

 

先使用下面的代码进行测试:

  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://dotnet.aspx.cc/SimpleCrossSiteRequests.aspx' ;  
  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 >   

 

然后,在服务器创建 CrossDomainRequest.aspx 的内容如下:

  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", "http://www.meng_xian_hui.com:801");  
  6.     Response.Write("孟宪会向各位朋友发来贺电:你的第一个跨域测试成功啦!!!");  
  7.   }  
  8. // --> </ mce:script >   

 

点击 “开始测试” 按钮,发送的请求和返回的响应信息如下:

 

  1. GET /SimpleCrossSiteRequests.aspx HTTP/1.1  
  2. Host: dotnet.aspx.cc  
  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: http://www.meng_xian_hui.com:801/CrossDomainAjax/SimpleCrossSiteRequests.html  
  11. Origin: http://www.meng_xian_hui.com: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: http://www.meng_xian_hui.com:801  
  18. Set-Cookie: ASP.NET_SessionId = wk5v5nrs5wbfi4rmpjy2jujb path =/; HttpOnly  
  19. Cache-Control: private  
  20. Content-Type: text/html; charset = utf -8  
  21. Content-Length: 84  

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

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

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

是不是现在就可以采用 XMLHttpRequest 来请求任意一个网站的数据呢?还是不行的。允许哪些域名可以访问,还需要服务器来设置 Access-Control-Allow-Origin 头来进行授权,具体的代码是:

Response.AddHeader("Access-Control-Allow-Origin", "http://www.meng_xian_hui.com:801");

这行代码就告诉浏览器,只有来自 http://www.meng_xian_hui.com:801 源下的脚本才可以进行访问。

好了,上面我们就完成了一个简单的跨域请求,怎么样?感觉还是不错的吧。下面我们进行一个“预检”请求。

 

 

 

注意:以下代码请在Firefox 3.5、Chrome 3.0、Safari 4之后的版本中进行测试。IE8的实现方法与其他浏览不同。

 

2,预检请求

预检请求首先需要向另外一个域名的资源发送一个 HTTP OPTIONS 请求头,其目的就是为了判断实际发送的请求是否是安全的。下面的2种情况需要进行预检:
a,不是上面的简单请求,比如使用Content-Type 为 application/xml 或 text/xml 的 POST 请求
b,在请求中设置自定义头,比如 X-JSON、X-MENGXIANHUI 等

 

注意:在 iis 里进行测试,必须在“应用程序扩展”里面配置 .aspx 扩展的动作允许 OPTIONS。

下面我们举一个预检的请求:

  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://dotnet.aspx.cc/PreflightedRequests.aspx' ;  
  13.   function crossDomainRequest() {  
  14.     document.getElementById("content").innerHTML  =  "开始进行请求……" ;  
  15.     if (xhr) {  
  16.       var xml  =  "<root>测试</root>" ;  
  17.       xhr.open('POST', url, true);  
  18.       xhr.setRequestHeader("POWERED-BY-MENGXIANHUI", "Approve");  
  19.       xhr.setRequestHeader("Content-Type", "application/xml");  
  20.       xhr.onreadystatechange  =  handler ;  
  21.       xhr.send(xml);  
  22.     } else {  
  23.     document.getElementById("content").innerHTML  =  "不能创建 XMLHttpRequest。" ;  
  24.     }  
  25.   }  
  26.   function handler(evtXHR) {  
  27.     if (xhr.readyState  == 4) {  
  28.       if (xhr.status  == 200) {  
  29.         var response  =  xhr .responseText;  
  30.         document.getElementById("content").innerHTML  =  "结果:"  + response;  
  31.       } else {  
  32.         document.getElementById("content").innerHTML  =  "不能进行跨越访问。" ;  
  33.       }  
  34.     }  
  35.     else {  
  36.       document.getElementById("content").innerHTML += "< br /> 执行状态 readyState:" + xhr.readyState;  
  37.     }  
  38.   }  
  39. // --> </ mce:script >   
  40. </ body >   
  41. </ html >   

上面的例子我们发送 xml 格式的数据,并且,发送一个非标准的HTTP头 POWERED-BY-MENGXIANHUI 来说明服务器端该如何设置响应头的。

在服务器端,PreflightedRequests.aspx 的内容如下:

  1. < %@ Page  Language = "C#"  % >   
  2. < mce:script   runat = "server" > <!--  
  3.   protected void Page_Load(object sender, EventArgs e)  
  4.   {  
  5.     if (Request.HttpMethod.Equals("GET"))  
  6.     {        
  7.       Response.Write("这个页面是用来测试跨域 POST 请求的,直接浏览意义不大。");  
  8.     }  
  9.     else if (Request.HttpMethod.Equals("OPTIONS"))  
  10.     {  
  11.       //通知客户端允许预检请求。并设置缓存时间  
  12.       Response.ClearContent();  
  13.       Response.AddHeader("Access-Control-Allow-Origin", "http://www.meng_xian_hui.com:801");  
  14.       Response.AddHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS");  
  15.       Response.AddHeader("Access-Control-Allow-Headers", "POWERED-BY-MENGXIANHUI");  
  16.       Response.AddHeader("Access-Control-Max-Age", "30");    
  17.       //此过程无需返回数据  
  18.       Response.End();        
  19.     }  
  20.     else if (Request.HttpMethod.Equals("POST"))  
  21.     {  
  22.       if (Request.Headers["Origin"].Equals("http://www.meng_xian_hui.com:801"))  
  23.       {  
  24.         System.Xml.XmlDocument doc  =  new  System.Xml.XmlDocument();  
  25.         doc.Load(Request.InputStream);  
  26.         Response.AddHeader("Access-Control-Allow-Origin", "http://www.meng_xian_hui.com:801");  
  27.         Response.Write("您提交的数据是:< br /> < br /> " + Server.HtmlEncode(doc.OuterXml));  
  28.       }  
  29.       else  
  30.       {  
  31.         Response.Write("不允许你的网站请求。");  
  32.       }  
  33.     }  
  34.   }  
  35. // --> </ mce:script >   

点击“开始测试”按钮,将会执行下面的一系列请求。

  1. OPTIONS /PreflightedRequests.aspx HTTP/1.1  
  2. Host: dotnet.aspx.cc  
  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. Origin: http://www.meng_xian_hui.com:801  
  11. Access-Control-Request-Method: POST  
  12. Access-Control-Request-Headers: powered-by-mengxianhui  
  13. HTTP/1.x 200 OK  
  14. Date: Sun, 10 Jan 2010 14:00:34 GMT  
  15. Server: Microsoft-IIS/6.0  
  16. X-Powered-By: ASP.NET  
  17. X-AspNet-Version: 2.0.50727  
  18. Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801  
  19. Access-Control-Allow-Methods: POST, GET, OPTIONS  
  20. Access-Control-Allow-Headers: POWERED-BY-MENGXIANHUI  
  21. Access-Control-Max-Age: 30  
  22. Set-Cookie: ASP.NET_SessionId = 5npqri55dl1k1zvij1tlw3re path =/; HttpOnly  
  23. Cache-Control: private  
  24. Content-Length: 0  
  25. POST /PreflightedRequests.aspx HTTP/1.1  
  26. Host: dotnet.aspx.cc  
  27. 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)  
  28. Accept: text/html,application/xhtml+xml,application/xml;q = 0 .9,*/*; q = 0 .8  
  29. Accept-Language: zh-cn,zh;q = 0 .5  
  30. Accept-Encoding: gzip,deflate  
  31. Accept-Charset: GB2312,utf-8;q = 0 .7,*; q = 0 .7  
  32. Keep-Alive: 300  
  33. Connection: keep-alive  
  34. POWERED-BY-MENGXIANHUI: Approve  
  35. Content-Type: application/xml; charset = UTF -8  
  36. Referer: http://www.meng_xian_hui.com:801/CrossDomainAjax/PreflightedRequests.html  
  37. Content-Length: 19  
  38. Origin: http://www.meng_xian_hui.com:801  
  39. Pragma: no-cache  
  40. Cache-Control: no-cache  
  41. < root > 测试 </ root >   
  42. HTTP/1.x 200 OK  
  43. Date: Sun, 10 Jan 2010 14:00:34 GMT  
  44. Server: Microsoft-IIS/6.0  
  45. X-Powered-By: ASP.NET  
  46. X-AspNet-Version: 2.0.50727  
  47. Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801  
  48. Set-Cookie: ASP.NET_SessionId = byvose45zmtbqy45d2a1jf2i path =/; HttpOnly  
  49. Cache-Control: private  
  50. Content-Type: text/html; charset = utf -8  
  51. Content-Length: 65  

 

以上的代码反映了预检请求的执行过程:首先发送 OPTIONS 请求头,用来向服务器咨询服务器的更多信息,以便为后续的真实请求做准备。比如是否支持 POST 方法等。值得注意的是:

浏览器还发送 Access-Control-Request-Method: POST 和 Access-Control-Request-Headers: powered-by-mengxianhui 请求头。

注意:以上过程是第一次请求的时候的过程,如果在 30 秒内重复点击按钮,你可以看不到 OPTIONS 这一过程。则执行过程是这样的:

 

  1. POST /PreflightedRequests.aspx HTTP/1.1  
  2. Host: dotnet.aspx.cc  
  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. POWERED-BY-MENGXIANHUI: Approve  
  11. Content-Type: application/xml; charset = UTF -8  
  12. Referer: http://www.meng_xian_hui.com:801/CrossDomainAjax/PreflightedRequests.html  
  13. Content-Length: 19  
  14. Origin: http://www.meng_xian_hui.com:801  
  15. Pragma: no-cache  
  16. Cache-Control: no-cache  
  17. < root > 测试 </ root >   
  18. HTTP/1.x 200 OK  
  19. Date: Sun, 10 Jan 2010 14:06:32 GMT  
  20. Server: Microsoft-IIS/6.0  
  21. X-Powered-By: ASP.NET  
  22. X-AspNet-Version: 2.0.50727  
  23. Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801  
  24. Set-Cookie: ASP.NET_SessionId = qs1c4urxywdbdx55u04pvual path =/; HttpOnly  
  25. Cache-Control: private  
  26. Content-Type: text/html; charset = utf -8  
  27. Content-Length: 65  

 

为什么会这样?细心的童鞋可能注意到了,在服务器端有一行代码 Response.AddHeader("Access-Control-Max-Age", "30");  它是用来设置预检的有效时间的,单位是秒。这一点要特别注意。

 

 

 

 

 

分享到:
评论

相关推荐

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

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

    详解AngularJS如何实现跨域请求

    跨域,前端开发中经常遇到的问题,AngularJS实现跨域方式类似于Ajax,使用CORS机制。 下面阐述一下AngularJS中使用$http实现跨域请求数据。 AngularJS XMLHttpRequest:$http用于读取远程服务器的数据 $http.post...

    vue+springboot实现项目的CORS跨域请求

    跨域资源共享CORS(Cross-origin Resource Sharing),是W3C的一个标准,允许浏览器向跨源的服务器发起XMLHttpRequest请求,克服ajax请求只能同源使用的限制。关于CORS的详细解读,可参考阮一峰大神的博客:跨域资源...

    js中跨域方法原理详解

    在js中,我们直接用XMLHttpRequest请求不同域上的数据时,是不可以的。但是,在页面上引入不同域上的js脚本文件却是可以的,jsonp正是利用这个特性来实现的。 比如,有个a.html页面,它里面的代码需要利用ajax获取一...

    「JavaScript」JS四种跨域方式详解

    浏览器都有一个同源策略,其限制之一就是第一种方法中我们说的不能通过ajax的方法去请求不同源中的文档。 它的第二个限制是浏览器中不同域的框架之间是不能进行js的交互操作的。 三、使用window.name来进行跨域 ...

    浅谈js中几种实用的跨域方法原理详解

    这里说的js跨域是指通过js在不同的域之间进行数据传输或通信,比如用ajax向一个不同的域请求数据,或者通过js获取页面中不同域的框架中(iframe)的数据。只要协议、域名、端口有任何一个不同,都被当作是不同的域。 ...

    跨域资源共享 CORS 详解

    它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从而克服了AJAX只能同源使用的限制。 本文详细介绍CORS的内部机制。 (图片说明:摄于阿联酋艾因(Al Ain)的绿洲公园) 一、简介 CORS需要浏览器和服务器同时...

    AJAX机制详解以及跨域通信

    1.Ajax 1.1.Ajax简介  Ajax简介这一部分我们主要是谈一下ajax的起源,ajax是什么?因为这些是跟技术无关的。... 这种新技术的主要目的就是为了使前端网页能够向服务器请求额外的数据而不需要卸载页面

    javascript跨域的4种方法和原理详解

    在js中,我们直接用XMLHttpRequest请求不同域上的数据时,是不可以的。但是,在页面上引入不同域上的js脚本文件却是可以的,jsonp正是利用这个特性来实现的。 比如,有个a.html页面,它里面的代码需要利用ajax获取一...

    javascript iframe跨域详解

    ajax本身实际上是通过XMLHttpRequest对象来进行数据的交互,而浏览器出于安全考虑,不允许js代码进行跨域操作,所以会警告。 2.有什么完美的解决方案么? 没有。解决方案有不少,但是只能是根据自己的实际情况来...

    Spring boot 和Vue开发中CORS跨域问题解决

    跨域资源共享CORS(Cross-origin Resource Sharing),是W3C的一个标准,允许浏览器向跨源的服务器发起XMLHttpRequest请求,克服ajax请求只能同源使用的限制。关于CORS的详细解读,可参考阮一峰大神的博客:跨域资源...

    python入门到高级全栈工程师培训 第3期 附课件代码

    02 jquery循环方法和attr,prop方法 03 jquery模态对话框与clone的应用 04 jqueryCSS操作之offsets,position以及scrolltop 05 jquery事件绑定与事件委托 06 jquery动画效果 07 jquery扩展与插件 08 jquery扩展补充 09...

Global site tag (gtag.js) - Google Analytics