1JS跨域访问解决方案总结0引言:跨域请求,顾名思义,就是一个站点中的资源去访问另外一个不同域名站点上的资源。这种情况很常见,比如说通过style标签加载外部样式表文件、通过img标签加载外部图片、通过script标签加载外部脚本文件、通过Webfont加载字体文件等等。默认情况下,脚本访问文档属性等数据采用的是同源策略(Sameoriginpolicy)。同源策略:如果两个页面的协议、域名和端口是完全相同的,那么它们就是同源的。同源策略是为了防止从一个地址加载的文档或脚本访问或者设置从另外一个地址加载的文档的属性。如果两个页面的主域名相同,则还可以通过设置document.domain属性将它们认为是同源的。随着Web2.0和SNS的兴起,Web应用对跨域访问的需求也越来越多,但在脚本中进行跨域请求是受安全性限制的,Web开发人员迫切需要提供一种更安全、方便的跨域请求方式来融合(Mashup)自己的Web应用。这样做的一个好处就是可以将请求分摊到不同的服务器,减轻单个服务器压力以提高响应速度;另外一个好处是可以将不同的业务逻辑分布到不同的服务器上以降低负载。值得庆幸的是,跨域请求的标准已经出台,主流浏览器也已经实现了这一标准。W3C工作组中的WebApplicationsWorkingGroup(Web应用工作组)发布了一个Cross-OriginResourceSharing(跨域资源共享规范)推荐规范来解决跨域请求的问题。该规范提供了一种更安全的跨域数据交换方法。具体规范的介绍可以访问上面提供的网站地址。值得注意的是:该规范只能应用在类似XMLHttprequest这样的API容器内。IE8、Firefox3.5及其以后的版本、Chrome浏览器、Safari4等已经实现了Cross-OriginResourceSharing规范,已经可以进行跨域请求了。一、支持跨域访问处理浏览器Cross-OriginResourceSharing的工作方式是通过添加HTTP头的方法来判断哪些资源允许Web浏览器访问该域名下的信息。然而,对于那些HTTP请求导致用户数据产生副作用的请求方法(特别是对于除了GET、某些MIME类型的POST之外的HTTP方法),该规范要求浏览器对请求进行“预先验”,通过发送HTTP的OPTIONS请求头询问服务器有哪些支持的方法,在征得服务器的同意后,再使用实际的HTTP请求方法发送实际的请求。服务器也可以通知客户端是否需要将验证信息(如Cookie和HTTPAuthentication数据)随同请求一起发送。下面我们就采用实际的例子说明Cross-OriginResourceSharing是如何工作的。1,简单请求2什么样的请求算是简单请求呢?简单请求必须满足下面2点:a,只使用GET、POST进行的请求,这里的POST只包括发送给服务器的数据类型(Content-Type)必须是application/x-、multipart/form-data或者text/plain中一个。b,HTTP请求没有设置自定义的请求头,如我们常用的X-JSON。先使用下面的代码进行测试:XML/HTML代码1.!DOCTYPEhtmlPUBLIC-//W3C//DTDXHTML1.0Transitional//EN2.=孟宪会之AJAX跨域请求测试/title6./head7.body8.inputtype='button'value='开始测试'onclick='crossDomainRequest()'/9.divid=content/div10.mce:scripttype=text/javascript!--11.varxhr=newXMLHttpRequest();12.varurl='(){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.functionhandler(evtXHR){324.if(xhr.readyState==4){25.if(xhr.status==200){26.varresponse=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:script37./body38./html然后,在服务器创建CrossDomainRequest.aspx的内容如下:C#代码1.%@PageLanguage=C#%2.mce:scriptrunat=server!--3.protectedvoidPage_Load(objectsender,EventArgse)4.{5.Response.AddHeader(Access-Control-Allow-Origin,);6.Response.Write(孟宪会向各位朋友发来贺电:你的第一个跨域测试成功啦!!!);7.}8.//--/mce:script点击“开始测试”按钮,发送的请求和返回的响应信息如下:XML/HTML代码1.GET/SimpleCrossSiteRequests.aspxHTTP/1.12.Host:dotnet.aspx.cc43.User-Agent:Mozilla/5.0(Windows;U;WindowsNT5.2;zh-CN;rv:1.9.1.7)Gecko/20091221Firefox/3.5.7(.NETCLR3.5.30729)4.Accept:text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.85.Accept-Language:zh-cn,zh;q=0.56.Accept-Encoding:gzip,deflate7.Accept-Charset:GB2312,utf-8;q=0.7,*;q=0.78.Keep-Alive:3009.Connection:keep-alive10.Referer:::Sun,10Jan201013:52:00GMT14.Server:Microsoft-IIS/6.015.X-Powered-By:ASP.NET16.X-AspNet-Version:2.0.5072717.Access-Control-Allow-Origin::ASP.NET_SessionId=wk5v5nrs5wbfi4rmpjy2jujb;path=/;HttpOnly19.Cache-Control:private20.Content-Type:text/html;charset=utf-821.Content-Length:84需要特别注意的是:在请求信息中,浏览器使用Origin这个HTTP头来标识该请求来自于;在返回的响应信息中,使用Access-Control-Allow-Origin头来控制哪些域名的脚本可以访问该资源。如果设置Access-Control-Allow-Origin:*,则允许所有域名的脚本访问该资源。如果有多个,则只需要使用逗号分隔开即可。注意:在服务器端,Access-Control-Allow-Origin响应头中的端口信息不能省略。有人可能会想:自己发送请求头会如何呢?比如xhr.setRequestHeader(Origin,);实践证明,自己设置Origin头是不行的。是不是现在就可以采用XMLHttpRequest来请求任意一个网站的数据呢?还是不行的。允许哪些域名可以访问,还需要服务器来设置Access-Control-Allow-Origin头来进行授权,具体的代码是:5Response.AddHeader(Access-Control-Allow-Origin,);这行代码就告诉浏览器,只有来自源下的脚本才可以进行访问。好了,上面我们就完成了一个简单的跨域请求,怎么样?感觉还是不错的吧。下面我们进行一个“预检”请求。2,预检请求预检请求首先需要向另外一个域名的资源发送一个HTTPOPTIONS请求头,其目的就是为了判断实际发送的请求是否是安全的。下面的2种情况需要进行预检:a,不是上面的简单请求,比如使用Content-Type为application/xml或text/xml的POST请求b,在请求中设置自定义头,比如X-JSON、X-MENGXIANHUI等注意:在iis里进行测试,必须在“应用程序扩展”里面配置.aspx扩展的动作允许OPTIONS。下面我们举一个预检的请求:XML/HTML代码1.!DOCTYPEhtmlPUBLIC-//W3C//DTDXHTML1.0Transitional//EN2.=孟宪会之AJAX跨域请求测试/title6./head7.body8.inputtype='button'value='开始测试'onclick='crossDomainRequest()'/9.divid=content/div10.mce:scripttype=text/javascript!--11.varxhr=newXMLHttpRequest();12.varurl='