注意:以下代码请在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. <htmlxmlns="http://www.w3.org/1999/xhtml">
  4. <head>
  5. <title>孟宪会之AJAX跨域请求测试</title>
  6. </head>
  7. <body>
  8. <inputtype='button'value='开始测试'onclick='crossDomainRequest()'/>
  9. <divid="content"></div>
  10. <mce:scripttype="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>
  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. <%@ PageLanguage="C#" %>
  2. <mce:scriptrunat="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. <%@ 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
  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 源下的脚本才可以进行访问。

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

AJAX(XMLHttpRequest)进行跨域请求方法详解(一)的更多相关文章

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

    AJAX(XMLHttpRequest)进行跨域请求方法详解(三) 2010年01月11日 08:48:00 阅读数:24213 注意:以下代码请在Firefox 3.5.Chrome 3.0.Saf ...

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

    注意:以下代码请在Firefox 3.5.Chrome 3.0.Safari 4之后的版本中进行测试.IE8的实现方法与其他浏览不同. 3,带验证信息的请求 身份验证是Web开发中经常遇到的问题,在跨 ...

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

    注意:以下代码请在Firefox 3.5.Chrome 3.0.Safari 4之后的版本中进行测试.IE8的实现方法与其他浏览不同. 2,预检请求 预检请求首先需要向另外一个域名的资源发送一个 HT ...

  4. 在ASP.NET 5应用程序中的跨域请求功能详解

    在ASP.NET 5应用程序中的跨域请求功能详解 浏览器安全阻止了一个网页中向另外一个域提交请求,这个限制叫做同域策咯(same-origin policy),这组织了一个恶意网站从另外一个网站读取敏 ...

  5. JS JSOP跨域请求实例详解

    JSONP(JSON with Padding)是JSON的一种“使用模式”,可用于解决主流浏览器的跨域数据访问的问题.这篇文章主要介绍了JS JSOP跨域请求实例详解的相关资料,需要的朋友可以参考下 ...

  6. JavaScript JSON AJAX 同源策略 跨域请求

    网页和Ajax和跨域的关系 1 Ajax使网页可以动态地.异步地的与服务器进行数据交互,可以让网页局部地与服务器进行数据交互 2 Ajax强调的是异步,但是会碰到跨域的问题. 3 而有很多技术可以解决 ...

  7. js中ajax如何解决跨域请求

    js中ajax如何解决跨域请求,在讲这个问题之前先解释几个名词 1.跨域请求 所有的浏览器都是同源策略,这个策略能保证页面脚本资源和cookie安全 ,浏览器隔离了来自不同源的请求,防上跨域不安全的操 ...

  8. VUE SpringCloud 跨域资源共享 CORS 详解

    VUE  SpringCloud 跨域资源共享 CORS 详解 作者:  张艳涛 日期: 2020年7月28日 本篇文章主要参考:阮一峰的网络日志 » 首页 » 档案 --跨域资源共享 CORS 详解 ...

  9. HTTP请求方法详解

    HTTP请求方法详解 请求方法:指定了客户端想对指定的资源/服务器作何种操作 下面我们介绍HTTP/1.1中可用的请求方法: [GET:获取资源]     GET方法用来请求已被URI识别的资源.指定 ...

随机推荐

  1. Java操作*.properties

    java中的properties文件是一种配置文件,主要用于表达配置信息,文件类型为*.properties,格式为文本文件,文件的内容是格式是"键=值"的格式,在properti ...

  2. 1.Perl 多线程:Threads

    详情可查看: perldoc threads 调用线程的方法: $thr = threads->create(FUNCTION, ARGS) #This will create a new th ...

  3. freemarker遍历list中的map

    前台: <select id="jq" name="jq" class="tsui" data-required="true ...

  4. do-while循环判断成绩的有效输入

    #include "stdio.h" void main() { int score; do { printf("请输入你的成绩(0-100):"); scan ...

  5. www.iis.net

    http://www.iis.net 这是一个神奇的网站 关于IIS的所有管理,在这里都能找到 今天,一个同事问我,  iis8 php的设置,一个环境变量的东西不知道怎么去设置, 然后我搜了下,在 ...

  6. 浅谈h5移动端页面的适配问题

    一.前言 昨天唠叨了哈没用的,今天说点有用的把.先说一下响应式布局吧,我一直认为响应式布局的分项目,一下布局简单得项目做响应式还是可以可以得.例如博客.后台管理系统等.但是有些会认为响应式很牛逼,尤其 ...

  7. CSS3秘笈:第七章

    第七章  margin.padding和border 1.盒模型:四个属性: (1)padding:内容与其边框线之间的空间. (2)border:盒子周围的直线 (3)background-colo ...

  8. Number-guessing Game

    Number-guessing Game Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 131072/65536K (Java/Othe ...

  9. jni中的参数含义

    #include <jni.h> JNIEXPORT jstring JNICALL Java_com_example_hellojni_MainActivity_helloFromC ( ...

  10. C#获取周的第一天、最后一天、月第一天和最后一天

    [csharp] view plaincopyprint? public class DateTimeTool { /// <summary> /// 获取指定日期所在周的第一天,星期天为 ...