- 浏览: 278727 次
- 性别:
- 来自: 上海
文章分类
最新评论
-
付小忠:
牛逼,解释到了点子上.
JAVA CAS原理深度分析 -
yhxf_ie:
csdn那些鬼转载都不注明出处的,这应该是原文了。
JAVA CAS原理深度分析 -
qq569349370:
终于找到一篇说得比较清楚的了,其他好多都是扰乱视听
JAVA CAS原理深度分析 -
lovemelong:
nice
JAVA CAS原理深度分析 -
Tyrion:
写的不错!
JAVA CAS原理深度分析
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。
先使用下面的代码进行测试:
- <!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://dotnet.aspx.cc/SimpleCrossSiteRequests.aspx' ;
- 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 >
然后,在服务器创建 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_hui.com:801");
- Response.Write("孟宪会向各位朋友发来贺电:你的第一个跨域测试成功啦!!!");
- }
- // --> </ mce:script >
点击 “开始测试” 按钮,发送的请求和返回的响应信息如下:
- GET /SimpleCrossSiteRequests.aspx HTTP/1.1
- Host: dotnet.aspx.cc
- 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 .8
- Accept-Language: zh-cn,zh;q = 0 .5
- Accept-Encoding: gzip,deflate
- Accept-Charset: GB2312,utf-8;q = 0 .7,*; q = 0 .7
- Keep-Alive: 300
- Connection: keep-alive
- Referer: http://www.meng_xian_hui.com:801/CrossDomainAjax/SimpleCrossSiteRequests.html
- Origin: http://www.meng_xian_hui.com:801
- HTTP/1.x 200 OK
- Date: Sun, 10 Jan 2010 13:52:00 GMT
- Server: Microsoft-IIS/6.0
- X-Powered-By: ASP.NET
- X-AspNet-Version: 2.0.50727
- Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801
- Set-Cookie: ASP.NET_SessionId = wk5v5nrs5wbfi4rmpjy2jujb ; path =/; HttpOnly
- Cache-Control: private
- Content-Type: text/html; charset = utf -8
- 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。
下面我们举一个预检的请求:
- <!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://dotnet.aspx.cc/PreflightedRequests.aspx' ;
- function crossDomainRequest() {
- document.getElementById("content").innerHTML = "开始进行请求……" ;
- if (xhr) {
- var xml = "<root>测试</root>" ;
- xhr.open('POST', url, true);
- xhr.setRequestHeader("POWERED-BY-MENGXIANHUI", "Approve");
- xhr.setRequestHeader("Content-Type", "application/xml");
- xhr.onreadystatechange = handler ;
- xhr.send(xml);
- } 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 >
上面的例子我们发送 xml 格式的数据,并且,发送一个非标准的HTTP头 POWERED-BY-MENGXIANHUI 来说明服务器端该如何设置响应头的。
在服务器端,PreflightedRequests.aspx 的内容如下:
- < %@ Page Language = "C#" % >
- < mce:script runat = "server" > <!--
- protected void Page_Load(object sender, EventArgs e)
- {
- if (Request.HttpMethod.Equals("GET"))
- {
- Response.Write("这个页面是用来测试跨域 POST 请求的,直接浏览意义不大。");
- }
- else if (Request.HttpMethod.Equals("OPTIONS"))
- {
- //通知客户端允许预检请求。并设置缓存时间
- Response.ClearContent();
- Response.AddHeader("Access-Control-Allow-Origin", "http://www.meng_xian_hui.com:801");
- Response.AddHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS");
- Response.AddHeader("Access-Control-Allow-Headers", "POWERED-BY-MENGXIANHUI");
- Response.AddHeader("Access-Control-Max-Age", "30");
- //此过程无需返回数据
- Response.End();
- }
- else if (Request.HttpMethod.Equals("POST"))
- {
- if (Request.Headers["Origin"].Equals("http://www.meng_xian_hui.com:801"))
- {
- System.Xml.XmlDocument doc = new System.Xml.XmlDocument();
- doc.Load(Request.InputStream);
- Response.AddHeader("Access-Control-Allow-Origin", "http://www.meng_xian_hui.com:801");
- Response.Write("您提交的数据是:< br /> < br /> " + Server.HtmlEncode(doc.OuterXml));
- }
- else
- {
- Response.Write("不允许你的网站请求。");
- }
- }
- }
- // --> </ mce:script >
点击“开始测试”按钮,将会执行下面的一系列请求。
- OPTIONS /PreflightedRequests.aspx HTTP/1.1
- Host: dotnet.aspx.cc
- 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 .8
- Accept-Language: zh-cn,zh;q = 0 .5
- Accept-Encoding: gzip,deflate
- Accept-Charset: GB2312,utf-8;q = 0 .7,*; q = 0 .7
- Keep-Alive: 300
- Connection: keep-alive
- Origin: http://www.meng_xian_hui.com:801
- Access-Control-Request-Method: POST
- Access-Control-Request-Headers: powered-by-mengxianhui
- HTTP/1.x 200 OK
- Date: Sun, 10 Jan 2010 14:00:34 GMT
- Server: Microsoft-IIS/6.0
- X-Powered-By: ASP.NET
- X-AspNet-Version: 2.0.50727
- Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801
- Access-Control-Allow-Methods: POST, GET, OPTIONS
- Access-Control-Allow-Headers: POWERED-BY-MENGXIANHUI
- Access-Control-Max-Age: 30
- Set-Cookie: ASP.NET_SessionId = 5npqri55dl1k1zvij1tlw3re ; path =/; HttpOnly
- Cache-Control: private
- Content-Length: 0
- POST /PreflightedRequests.aspx HTTP/1.1
- Host: dotnet.aspx.cc
- 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 .8
- Accept-Language: zh-cn,zh;q = 0 .5
- Accept-Encoding: gzip,deflate
- Accept-Charset: GB2312,utf-8;q = 0 .7,*; q = 0 .7
- Keep-Alive: 300
- Connection: keep-alive
- POWERED-BY-MENGXIANHUI: Approve
- Content-Type: application/xml; charset = UTF -8
- Referer: http://www.meng_xian_hui.com:801/CrossDomainAjax/PreflightedRequests.html
- Content-Length: 19
- Origin: http://www.meng_xian_hui.com:801
- Pragma: no-cache
- Cache-Control: no-cache
- < root > 测试 </ root >
- HTTP/1.x 200 OK
- Date: Sun, 10 Jan 2010 14:00:34 GMT
- Server: Microsoft-IIS/6.0
- X-Powered-By: ASP.NET
- X-AspNet-Version: 2.0.50727
- Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801
- Set-Cookie: ASP.NET_SessionId = byvose45zmtbqy45d2a1jf2i ; path =/; HttpOnly
- Cache-Control: private
- Content-Type: text/html; charset = utf -8
- Content-Length: 65
以上的代码反映了预检请求的执行过程:首先发送 OPTIONS 请求头,用来向服务器咨询服务器的更多信息,以便为后续的真实请求做准备。比如是否支持 POST 方法等。值得注意的是:
浏览器还发送 Access-Control-Request-Method: POST 和 Access-Control-Request-Headers: powered-by-mengxianhui 请求头。
注意:以上过程是第一次请求的时候的过程,如果在 30 秒内重复点击按钮,你可以看不到 OPTIONS 这一过程。则执行过程是这样的:
- POST /PreflightedRequests.aspx HTTP/1.1
- Host: dotnet.aspx.cc
- 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 .8
- Accept-Language: zh-cn,zh;q = 0 .5
- Accept-Encoding: gzip,deflate
- Accept-Charset: GB2312,utf-8;q = 0 .7,*; q = 0 .7
- Keep-Alive: 300
- Connection: keep-alive
- POWERED-BY-MENGXIANHUI: Approve
- Content-Type: application/xml; charset = UTF -8
- Referer: http://www.meng_xian_hui.com:801/CrossDomainAjax/PreflightedRequests.html
- Content-Length: 19
- Origin: http://www.meng_xian_hui.com:801
- Pragma: no-cache
- Cache-Control: no-cache
- < root > 测试 </ root >
- HTTP/1.x 200 OK
- Date: Sun, 10 Jan 2010 14:06:32 GMT
- Server: Microsoft-IIS/6.0
- X-Powered-By: ASP.NET
- X-AspNet-Version: 2.0.50727
- Access-Control-Allow-Origin: http://www.meng_xian_hui.com:801
- Set-Cookie: ASP.NET_SessionId = qs1c4urxywdbdx55u04pvual ; path =/; HttpOnly
- Cache-Control: private
- Content-Type: text/html; charset = utf -8
- Content-Length: 65
为什么会这样?细心的童鞋可能注意到了,在服务器端有一行代码 Response.AddHeader("Access-Control-Max-Age", "30"); 它是用来设置预检的有效时间的,单位是秒。这一点要特别注意。
相关推荐
3. **发送的是XHR(XMLHttpRequest)请求**:即使用了Ajax技术进行异步请求。 当这三个条件同时满足时,浏览器就会阻止请求的完成,导致前端无法正常处理后端的响应数据。 #### 二、解决方案详解 针对上述问题,...
3. **AJAX请求不能发送**:非同源的AJAX请求将被禁止,这意味着JavaScript无法通过XMLHttpRequest对象向非同源服务器发送请求。 尽管这些限制是出于安全考虑,但在某些场景下也会带来不便,比如合理的跨域数据交换...
【Ajax跨域文件上传详解】 Ajax(Asynchronous JavaScript and XML)技术允许我们在不刷新整个页面的情况下,通过JavaScript异步地与服务器交换数据并更新部分网页内容。在现代Web开发中,Ajax广泛应用于创建动态和...
6. **XMLHttpRequest Level 2:** 支持跨域的Ajax请求,所有现代浏览器都已支持。 7. **WebSocket:** 实时双向通信,支持跨域。 8. **Server-Sent Events(SSE):** 服务器向客户端单向推送数据。 9. **后台代理:...
跨域资源共享CORS(Cross-origin Resource Sharing),是W3C的一个标准,允许浏览器向跨源的服务器发起XMLHttpRequest请求,克服ajax请求只能同源使用的限制。关于CORS的详细解读,可参考阮一峰大神的博客:跨域资源...
### Ajax异步(请求)提交类支持跨域 #### 概述 在现代Web开发中,Ajax技术的应用非常广泛,它允许网页不刷新整个页面的情况下与服务器进行数据交互,极大地提升了用户体验。然而,Ajax默认是不允许跨域请求的,这是...
### JavaScript 跨域请求包装函数与用法详解 在现代Web开发中,跨域问题是一个常见的挑战。由于浏览器出于安全考虑实施了同源策略(Same-Origin Policy),这限制了一个域下的文档或脚本如何与另一个域下的资源进行...
**Ajax 远程跨域获取 163RSS 新闻技术详解** Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个网页的情况下,能够更新部分网页的技术。它通过JavaScript与服务器进行异步数据交换,使用户体验更加流畅...
### JavaScript源代码:Ajax解决跨域之设置CORS响应头实现跨域案例详解 #### 跨源资源共享(CORS)概述 跨源资源共享(Cross-Origin Resource Sharing,简称CORS)是一种安全机制,允许Web应用程序从不同的域名...
- **安全问题**:由于Ajax请求通常局限于同源策略,跨域请求需要服务器端配合。 - **SEO难题**:搜索引擎爬虫无法执行JavaScript,可能影响内容的抓取。 - **用户体验**:若处理不当,可能导致用户界面卡顿或失去...
2. **使用$.ajax()方法发起跨域请求**: ```javascript $(function () { $.ajax({ type: 'get', url: 'http://oa03.bankoftianjin.com/app/tcwmain.nsf/ag_OutputAllProDataCountByID?openagent&proid=dbsy,...
本篇文章详细分析了自定义ajax支持跨域组件封装相关的知识点,包括Class.create()分析、Object.extend()分析、Ajax定义、XMLHttpRequest详解、readystatechange事件、open方法、send方法、ajax跨域和解决跨域问题等...
此外,还有一些其他的选项,如`contentType`(请求的Content-Type头)、`crossDomain`(是否跨域请求)、`xhrFields`(自定义XMLHttpRequest实例字段)等,它们进一步增强了$.ajax()方法的灵活性。 使用$.ajax()...
1. **基础概念**:解释Ajax的工作原理,包括异步通信、 XMLHttpRequest对象(XHR)的作用以及如何通过JavaScript与服务器进行交互。 2. **XML与JSON**:由于Ajax最初与XML紧密相关,因此会介绍XML的基本结构和解析...
然而,这只是一个简化版,实际的jQuery `$.ajax()`方法还包含更多的功能,如错误处理、跨域请求的CORS支持、预处理和后处理函数、以及更复杂的序列化和反序列化逻辑。在生产环境中,我们可能需要考虑这些问题以提高...
**Ajax 客户端 XMLHttpRequest 实例详解** Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个页面的情况下,能够更新部分网页的技术。它通过在后台与服务器进行少量数据交换,使得网页实现了异步更新。...
**Ajax(Asynchronous JavaScript and XML)技术详解** Ajax,全称异步JavaScript和XML,是一种在无需刷新整个网页的情况下,能够更新部分网页的技术。自2005年被广泛使用以来,它极大地改善了用户体验,使得网页...