Angular经过CORS实现跨域方案

 之前有一篇很老的文章网上转了不少,包括如今若是你百度"跨域"这个关键字,前几个推荐的都是"Javascript跨域总结与解决方案".看了一下感受手段有点陈旧了,有一些好比document.domain还有iframe的解决方案委实"丑陋"一些,感受再也不适用于如今一些项目中.前端

    就拿iframe来讲做为一个前端工程师,我极为讨厌iframe这种东西.它不光增长了性能上的高负荷,同时也不利于掌控.ajax

    在Angular应用中实现跨域的方式相对简单,基本上经过两种方式便可.一种是JSONP,另外一种是经过CORS.前者是相对比较老的手法,后者我感受更加给力一点,因此本文主要说一下Angular如何与CORS配合跨域.json

    能不使用JSONP就尽可能不使用,这是着手于Angular跨域的一个原则吧.无论怎么说,script的标签嵌入感受仍是low了点.后端

    Angular推崇的时先后端分离,因此跨域由哪一方实现成为一个问题.这个就不得不说前端技术上的局限性,即便是相对好用的JSONP对于非GET请求也是无能为力的,由于它本质上仍是经过script去get一些资源.api

    JSONP这种只能GET的限制,在Angular推崇RESTful风格接口的API场景下,就彻底制约了它的使用,总不能弃POST和PUT那些无论.而且JSONP的错误处理很弱,不尽人意.总以前端实现跨域都有各类各样的局限性,又好比像document.domain则只能用于主域相同,子域不一样的状况.跨域

    因此总结而言,虽然前端有多种方式处理跨域,可是多而不精,缺点都比较明显.相对而言更好的方式是经过后端参与处理,这样作不只适用性更强,同时前端只要发送正常的Ajax请求便可.这样的技术叫作CORS.浏览器

    Cross-Origin Resource Sharing跨域资源共享,应该算是如今最为推荐的跨域处理方案.不只适用于各类Method,并且更加方便和简单.固然了,这么吊的东西只有现代浏览器支持,IE8一下的老古董就不要想了.服务器

 

    CORS实现原理前端工程师

 

    虽然经过CORS实现跨域基本上彻底由后端实现,不过身为一个给力的前端.仍是要掌握一下这一原理,以便当你遇到不靠谱的后端时,不至于...你懂得app

    CORS的本质让服务器经过新增响应头Access-Control-Allow-Origin,经过HTTP方式来实现资源共享,让每一个请求的服务直接返回资源.它使用了HTTP交互方式来肯定请求源是否有资格请求该资源,而且经过设置HTTP Header来控制访问资源的权限.

    具体的过程是这样的前端发送一个正常的请求:

$http.get('www.cros.com/api/data',{params:{

  name: '顽Shi'

}})

    后端设置一下response的header:

 

Access-Control-Allow-Origin: "*"

Access-Control-Allow-Methods: "GET"

Access-Control-Max-Age: "60"   

    而后你观察一下浏览器的行为会发现有趣的事,浏览器在没有你干预的状况下,发现这是一个跨域请求.因此它没有直接发送GET请求,而是发送了一个OPTIONS请求询问是否能够跨域访问该资源,这个过程咱们能够称之为"预检".

    而后咱们看到OPTIONS的response返回了相似下面的信息:

HTTP/1.1 200 OK

 

Date: Mon, 01 Dec 2013 01:15:39 GMT

Server: Apache/2.0.61 (Unix)

Access-Control-Allow-Origin: *

Access-Control-Allow-Methods: GET

Access-Control-Max-Age: 60

Content-Encoding: gzip

Content-Length: 0

Connection: Keep-Alive

Content-Type: text/text

    这里的这几个Access头的内容就是服务器后端加上去的,它告诉了浏览器此后的60秒内,全部域均可以经过GET方法进行跨域访问该资源.而后浏览器自动再次发送了真正的GET请求,并返回对应的结果.

    注意这一过程是浏览器自动实现的,这一点是否是很是棒.一些header信息的设置以下:

Access-Control-Allow-Origin:| * // 受权的源控制

Access-Control-Max-Age:// 受权的时间

Access-Control-Allow-Credentials: true | false // 控制是否开启与Ajax的Cookie提交方式

Access-Control-Allow-Methods:[,]* // 容许请求的HTTP Method

Access-Control-Allow-Headers:[,]* // 控制哪些header能发送真正的请求    

    这里还有一处须要前端工程师协做的地方就是cookie的传递,默认状况下经过CORS这样的方式是不会传递cookie.通常强制性将cookie添加到header的作法,也会被浏览器拒绝并报错.上面看到了在服务器端会经过添加一个response头,Access-Control-Allow-Credentials来控制是否容许Cookie的提交.

    在Angular中咱们须要进行一些设置达到目的:

$http.post(url, {withCredentials: true, ...})

// 或者

$http({withCredentials: true, ...}).post(...)

// 或者

.config(function ($httpProvider) {

  $httpProvider.defaults.withCredentials = true;

}

    若是是jQuery则要设置以下:

$.ajax("www.cros.com/api/data", {

  type: "GET",

  xhrFields: {

    withCredentials: true

  },

  crossDomain: true,

  success: function(data, status, xhr) {

  }

});

    CORS的过程描述完毕,在网上找到一张图片:

 

    CORS的分类

    若是仔细观察浏览器的行为会发现,并非全部的跨域请求都会发送OPTIONS请求.是否是有些奇怪,这就涉及到CORS的分类,简单请求和复杂请求.

    HTTP的header一般包含下面这些内容:

Accept

Accept-Language

Content-Language

Last-Event-ID

Content-Type的值仅是下列之一:

                                              application/x-www-form-urlencoded

                                              multipart/form-data

                                              text/plain

    HTTP方法是HEAD,GET,POST之一,同时HTTP的header包含如上面所示.任何一个不知足这两种要求的请求,都是复杂请求.好比发送PUT,DELETE等HTTP动做,或者Content-Type: application/json的内容.

    只有复杂请求包含"预检"这一动做,另外Access-Control-Max-Age应该也会影响OPTIONS请求的发送.

相关文章
相关标签/搜索