• 浏览器跨域


    相关问题

    1. 什么是跨域
    2. 为什么会跨域
    3. 为什么会有跨域的限制
    4. 怎么解决跨域

    回答关键点

    CORS同源策略

    跨域问题的来源是浏览器为了请求安全而引入的基于同源策略的安全特性。当页面和请求的协议、主机名或端口不同时,浏览器判定两者不同源,即为跨域请求。需要注意的是跨域是浏览器的限制,服务端并不受此影响。当产生跨域时,我们可以通过 JSONP、CORS、postMessage 等方式解决。

    知识点深入

    跨域问题的来源
    跨域问题的来源是浏览器为了请求安全而引入的基于同源策略Same-origin policy)的安全特性。同源策略是浏览器一个非常重要的安全策略,基于这个策略可以限制非同源的内容与当前页面进行交互,从而减少页面被攻击的可能性
    当页面和请求的协议、主机名或端口不同时,浏览器判定两者不同源,从而产生跨域。需要注意的是跨域是浏览器的限制,实际请求已经正常发出和响应了

    如何判定跨域
    在这里插入图片描述

    如上图所示,一个 origin 由协议(Protocol)、主机名(Host)和端口(Port)组成,这三块也是同源策略的判定条件,只有当协议、主机名和端口都相同时,浏览器才判定两者是同源关系,否则即为跨域。

    跨域的解决方案

    前端常见的跨域解决方案有 CORS、反向代理(Reverse Proxy)、JSONP

    • CORS (Cross-Origin Resource Sharing)
      CORS 是目前最为广泛的解决跨域问题的方案。方案依赖服务端/后端在响应头中添加 Access-Control-Allow-* 头,告知浏览器端通过此请求。

    • 涉及到的端

      CORS 只需要服务端/后端支持即可,不涉及前端改动。

    • 具体实现方式

      CORS 将请求分为简单请求(Simple Requests)和需预检请求(Preflighted requests),不同场景有不同的行为:

    • 简单请求

      不会触发预检请求的称为简单请求。当请求满足以下条件时就是一个简单请求:

      • 请求方法:GET、HEAD、POST
      • 请求头:Accept、Accept-Language、Content-Language、Content-Type
      • Content-Type 仅支持:application/x-www-form-urlencoded、multipart/form-data、text/plain
    • 需预检请求

      当一个请求不满足以上简单请求的条件时,浏览器会自动向服务端发送一个 OPTIONS 请求,通过服务端返回的 Access-Control-Allow-* 判定请求是否被允许。

    • CORS 引入了以下几个以 Access-Control-Allow-* 开头:

      • Access-Control-Allow-Origin 表示允许的来源
      • Access-Control-Allow-Methods 表示允许的请求方法
      • Access-Control-Allow-Headers 表示允许的请求头
      • Access-Control-Allow-Credentials 表示允许携带认证信息
        当请求符合响应头的这些条件时,浏览器才会发送并响应正式的请求。
    • 反向代理
      反向代理解决跨域问题的方案依赖同源的服务端对请求做一个转发处理,将请求从跨域请求转换成同源请求。

    • 涉及到的端

      反向代理只需要服务端/后端支持,几乎不涉及前端改动,只用切换接口即可。

    • 具体实现方式

      反向代理的实现方式为在页面同域下配置一套反向代理服务,页面请求同域的服务端,服务端请求上游的实际的服务端,之后将结果返回给前端。

    • JSONP
      JSONP 是一个相对古老的跨域解决方案,只支持 GET 请求。主要是利用了浏览器加载 JavaScript 资源文件时不受同源策略的限制而实现跨域获取数据。

    • 涉及到的端

      JSONP 需要服务端和前端配合实现。

    • 具体实现方式

      JSONP 的原理是利用了浏览器加载 JavaScript 资源文件时不受同源策略的限制而实现的。具体流程如下:

    • 全局注册一个函数,例如:window.getHZFEMember = (num) => console.log('HZFE Member: ’ + num);。

    • 构造一个请求 URL,例如:https://hzfe.org/api/hzfeMember?callback=getHZFEMember。

    • 生成一个