在我们日常的项目开发时使用AJAX,传统的Ajax请求只能获取在同一个域名下面的资源,但是HTML5打破了这个限制,允许Ajax发起跨域的请求。浏览器是可以发起跨域请求的,比如你可以外链一个外域的图片或者脚本。但是Javascript脚本是不能获取这些资源的内容的,它只能被浏览器执行或渲染。主要原因还是出于安全考虑,浏览器会限制脚本中发起的跨站请求。(同源策略, 即JavaScript或Cookie只能访问同域下的内容)。跨域的解决方案有多重JSONP、Flash、Iframe等,当然还有CORS(跨域资源共享,Cross-Origin Resource Sharing)今天就来了解下CORS的原理,以及如何使用。

一、CORS概述

跨源资源共享标准通过新增一系列 HTTP 头,让服务器能声明那些来源可以通过浏览器访问该服务器上的各类资源(包括CSS、图片、JavaScript 脚本以及其它类资源)。另外,对那些会对服务器数据造成破坏性影响的 HTTP 请求方法(特别是 GET 以外的 HTTP 方法,或者搭配某些MIME类型的POST请求),标准强烈要求浏览器必须先以 OPTIONS 请求方式发送一个预请求(preflight request),从而获知服务器端对跨源请求所支持 HTTP 方法。在确认服务器允许该跨源请求的情况下,以实际的 HTTP 请求方法发送那个真正的请求。服务器端也可以通知客户端,是不是需要随同请求一起发送信用信息(包括 Cookies 和 HTTP 认证相关数据)。

二、CORS原理

例如:域名A(http://a.example)的某 Web 应用程序中通过<img>标签引入了域名B(http://b.foo)站点的某图片资源(http://b.foo/image.jpg)。这就是一个跨域请求,请求http报头包含Origin: http://a.example,如果返回的http报头包含响应头 Access-Control-Allow-Origin: http://a.example (或者Access-Control-Allow-Origin: http://a.example),表示域名B接受域名B下的请求,那么这个图片就运行被加载。否则表示拒绝接受请求。

三、CORS跨域请求控制方法

1.http请求头

Origin: 普通的HTTP请求也会带有,在CORS中专门作为Origin信息供后端比对,表明来源域。

Access-Control-Request-Method: 接下来请求的方法,例如PUT, DELETE等等

Access-Control-Request-Headers: 自定义的头部,所有用setRequestHeader方法设置的头部都将会以逗号隔开的形式包含在这个头中

2.http响应头

然后浏览器再根据服务器的返回值判断是否发送非简单请求。简单请求前面讲过是直接发送,只是多加一个origin字段表明跨域请求的来源。然后服务器处理完请求之后,会再返回结果中加上如下控制字段

Access-Control-Allow-Origin: 允许跨域访问的域,可以是一个域的列表,也可以是通配符"*"。这里要注意Origin规则只对域名有效,并不会对子目录有效。即http://foo.example/subdir/ 是无效的。但是不同子域名需要分开设置,这里的规则可以参照同源策略

Access-Control-Allow-Credentials: 是否允许请求带有验证信息,XMLHttpRequest请求的withCredentials标志设置为true时,认证通过,浏览器才将数据给脚本程序。

Access-Control-Expose-Headers: 允许脚本访问的返回头,请求成功后,脚本可以在XMLHttpRequest中访问这些头的信息

Access-Control-Max-Age: 缓存此次请求的秒数。在这个时间范围内,所有同类型的请求都将不再发送预检请求而是直接使用此次返回的头作为判断依据,非常有用,大幅优化请求次数

Access-Control-Allow-Methods: 允许使用的请求方法,以逗号隔开

Access-Control-Allow-Headers: 允许自定义的头部,以逗号隔开,大小写不敏感

四、浏览器支持情况

在大部分现代浏览器中有所支持,支持(部分支持)CORS协议的浏览器有IE8+, Firefox5+, Chrome12+, Safari4+,移动端几乎全支持。

注:Internet Explorer 8 、9使用 XDomainRequest 对象实现CORS。

五、CORS使用案例

案例环境:客户端使用jQuery,服务端WebApi(2.2)。因本人使用.net语言,所以服务端就使用webApi来演示了。

首先新建一个webApi项目,这里就不截图一步步介绍了,然后使用Nuget安装支持cors的扩展组件,

Install-Package Microsoft.AspNet.WebApi.Cors

然后打开App_Start问价夹下的WebConfig.cs配置文件类,在Register方法中配置一个全局的cors,为了方便我将一些参数配置到web.config配置文件中

如果不想使用全局的CORS,可以在某个方法或者ApiController上这样配置:[EnableCors(origins: "*", headers: "*", methods: "*")],可以使用具体的参数,多个参数以逗号分隔,不用说,肯定英文逗号。origins 域名要带上http的顶级域名。需要添加 using System.Web.Http.Cors;

一般请求来说,客户端的AJAX请求不需要做任何改变,只需要服务端稍作改变即可。

客户端js代码:     apiRootPath是我预先设置的api的顶级域名。

view sourceprint?
01.$.ajax({
02.url: apiRootPath + "api/Account/Register",
03.type: "post",
04.data: {
05."UserName": mobile,
06."Pass<a href="http://www.it165.net/edu/ebg/" target="_blank" class="keylink">word</a>": pwd
07.},
08.dataType: "json",
09.success: function (data) {
10.if (data.State == true) {
11.RegSuccess(mobile, pwd);
12.else {
13.$("#errorText").html(data.Message);
14.$("#registerBtn").text("注册");
15.}
16.}
17.});

因为我配置了全局的CORS方法,而且服务端没有特别之处了,和普通的网站(不跨越)写法一致,这里就不予贴出了。

如果需要对请求进行身份验证,怎么办?我们一cookies实现这个验证。

view sourceprint?
01.$.ajax({
02.url: apiRootPath + "api/Account/Login",
03.type: "post",
04.data: {
05."UserName": userName,
06."Pass<a href="http://www.it165.net/edu/ebg/" target="_blank" class="keylink">word</a>": password
07.},
08.crossDomain: true,
09.xhrFields: {
10.withCredentials: true
11.},
12.dataType: "json",
13.success: function (data) {
14.if (data.State == true) {
15.MLogin(userName, password);
16.else {
17.$("#loginBtn").text("登录");
18.$("#errorText").html(data.Message);
19.}
20.}
21.});

注意这个两句话:crossDomain: true,xhrFields: {withCredentials: true}

六:安全隐患

如果程序猿偷懒将Access-Control-Allow-Origin设置为:Access-Control-Allow-Origin: * 允许任何来自任意域的跨域请求,那么久存在被 DDoS攻击的可能。

and待补充。。。

转载于:https://www.cnblogs.com/liuchuanfeng/p/6833166.html

跨域的另一种解决方案CORS(CrossOrigin Resource Sharing)跨域资源共享相关推荐

  1. CORS(Cross-Origin Resource Sharing)跨域资源共享

    CORS(Cross-Origin Resource Sharing)跨域资源共享 由于浏览器同源策略的影响,跨域问题在我们日常开发中经常遇到,此前有一系列的跨域问题的解决方案,如JSONP.flas ...

  2. Spring Boot中对CORS(Cross-Origin Resource Sharing 跨域资源共享)的支持

    1.概念 Cors(Cross-Origin Resourece Sharing)是由W3C制定的一种跨域资源共享技术标准,目的是为了解决前端的跨域请求. 2.实验步骤 SpringBoot中配置Co ...

  3. CORS(Cross-Origin Resource Sharing) 跨域资源共享

    CORS 全称:Cross-Origin Resource Sharing 中文意思:跨域资源共享? 好吧,目前中文方面的资料还比较少,能搜索到的那仅有的几篇相关介绍,也几乎是雷同的. 最近工作上也有 ...

  4. 跨域的另一种解决方案——CORS(Cross-Origin Resource Sharing)跨域资源共享

    跨域的另一种解决方案--CORS(Cross-Origin Resource Sharing)跨域资源共享 参考文章: (1)跨域的另一种解决方案--CORS(Cross-Origin Resourc ...

  5. 跨域问题的4种解决方案

    更多内容,请访问我的 个人博客. 前言 难以置信,我居然被跨域问题折磨了一上午.相信很多程序员都遇到过跨域问题,当然,解决方案也有很多种.但我今天尝试了无数种办法,依旧没有解决.直到最后我媳妇儿给了我 ...

  6. 跨域问题的几种解决方案汇总

    来源:github.com/campcc/blog 什么是"跨源" JSONP CORS 兼容性 实现原理 请求类型 PostMessage Websocket Nginx 反向代 ...

  7. 前端跨域问题的几种解决方案

    前端跨域问题的几种解决方案 参考文章: (1)前端跨域问题的几种解决方案 (2)https://www.cnblogs.com/xinxingyu/p/6075881.html 备忘一下.

  8. js跨域解决方案php,详解js跨域原理以及2种解决方案_javascript技巧

    1.什么是跨域 我们经常会在页面上使用ajax请求访问其他服务器的数据,此时,客户端会出现跨域问题. 跨域问题是由于javascript语言安全限制中的同源策略造成的. 简单来说,同源策略是指一段脚本 ...

  9. 解决ajax跨域问题【5种解决方案】

    什么是跨域问题? 跨域问题来源于JavaScript的"同源策略",即只有 协议+主机名+端口号 (如存在)相同,则允许相互访问.也就是说JavaScript只能访问和操作自己域下 ...

最新文章

  1. java之final
  2. 问号和星号 php,星号和问号两个通配符
  3. NHibernate使用时,不能返回自己的异常的解决办法
  4. 推荐大家一个保护视力的Visual Studio皮肤
  5. 数据结构实验之栈与队列三:后缀式求值
  6. 撸过一万行代码,你看过这篇文章吗?
  7. python turtle应用实例_turtle库应用实例2-六芒星的绘制
  8. python 内置标准库socketserver模块的思考
  9. python逻辑运算符不懂_Python之逻辑运算符
  10. ASP.NET自定义错误页面,分离配置信息,多环境发布
  11. linux的abrt目录满了,linux:abrt-cli list
  12. Android源码目录结构,kotlin中文文档pdf
  13. 明解C语言。初级版 部分练习代码
  14. 数据库范式之间的转换 - 保持函数依赖分解与有/无损分解
  15. Proteus,keil5仿真运行stm32程序,流水灯详细教程
  16. linux 系统速度慢,Linux运维人员你知道Linux系统运行速度太慢的原因吗?
  17. 查看计算机硬件配置方法(包括内存条、显卡型号,卡槽还剩几个)
  18. ImageIO的图片压缩算法
  19. 免费AI改图神器,一个万能宝藏在线工具箱
  20. [案例3-1]基于控制台的购书系统

热门文章

  1. 关于软件测试从业者必备知识,录了几个视频
  2. Latex学习笔记(二)各标题的大小设置
  3. 从NTP网络授时到社交GPS千奇百怪的应用方式
  4. 2021-03-19腾飞XXEachXX-00.00-N.009
  5. HTML5游戏开发(四)
  6. .NET Framework
  7. 《红岩》小说人物关系
  8. mysql sql批量插入_mysql批量插入数据的纯sql脚本
  9. DB2的数据库备份与恢复
  10. Revit中计算管道的总长--折算管件