当前位置:首页 > 技术文章 > 正文内容

如何解决前后端分离的跨域问题?

arlanguage4个月前 (12-26)技术文章36

跨域问题是由于浏览器的一种安全策略而出现的,即同源策略(Same origin policy。同源策略由 Netscape 提出,是浏览器最核心也是最基本的安全功能。所谓“同源”指的是:

协议、域名、端口号都相同,只要有一个不相同,那么都是非同源。

同源:协议、域名、端口号一致

出于安全的考虑,默认禁止跨域访问。如果跨域访问,这时候通常就会报错:

has been blocked by CORS policy: No 'Access-Control-Allow-Origin'

CORS全称是"跨域资源共享"(Cross-origin resource sharing),这是一个W3C标准。利用CORS可以允许浏览器向跨源服务器发出XMLHttpRequest请求,从而克服了AJAX只能同源使用的限制。

什么是CORS

解决跨域问题除了上述CORS以外,还有JSONP、iframe跨域、代理等办法。本文重点介绍CORS这种最为通用的解决办法,相比JSONP只能支持GET请求,CORS可以支持所有类型的HTTP请求。

CORS跨域请求原理

CORS通信需要浏览器和服务器同时支持。整个CORS通信过程,都是浏览器自动完成,对于开发者来说,CORS通信与同源的AJAX通信没有差别,代码完全一样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息。实现CORS通信的关键是服务器,需要服务器实现CORS接口,方可跨源通信。

浏览器将CORS请求分成两类:简单请求(simple request)非简单请求(not-so-simple request)

简单请求

对于简单请求,浏览器发出CORS请求,在头信息中增加一个Origin字段。

GET /cors HTTP/1.1
Origin: http://api.weishuo.com
Host: api.alice.com
Accept-Language: en-US
Connection: keep-alive
User-Agent: Mozilla/5.0...

Origin字段用来说明,本次请求来自哪个源(协议 + 域名 + 端口)。服务器根据这个值,决定是否同意这次请求。如果Origin指定的域名在许可范围内,服务器返回的响应,会多出几个头信息字段。

Access-Control-Allow-Origin: http://api.weishuo.com
Access-Control-Allow-Credentials: true
Access-Control-Expose-Headers: FooBar
Content-Type: text/html; charset=utf-8

其中 Access-Control-Allow-Origin 是必须的。如果我们要放行所有跨域请求,那么可以在 Servlet 返回响应时,添加如下代码:

resp.setHeader("Access-Control-Allow-Origin", "*");

其中*表示支持所有网站访问,也可以为 Access-Control-Allow-Origin 配置专门域名。

非简单请求

CORS非简单请求过程

非简单请求的CORS请求过程如上图所示,在正式通信之前,会增加一次HTTP查询请求,称为预检请求

OPTIONS /cors HTTP/1.1
Origin: http://api.weishuo.com
Access-Control-Request-Method: PUT
Access-Control-Request-Headers: X-Custom-Header
Host: api.alice.com
Accept-Language: en-US
Connection: keep-alive
User-Agent: Mozilla/5.0...

预检请求用的请求方法是 OPTIONS,表示这个请求是用来询问的。头信息里面,关键字段是 Origin,表示请求来自哪个源。

服务器收到"预检"请求以后,检查了OriginAccess-Control-Request-Method Access-Control-Request-Headers 字段以后,确认允许跨源请求,就可以做出预检响应

HTTP/1.1 200 OK
Date: Mon, 01 Dec 2008 01:15:39 GMT
Server: Apache/2.0.61 (Unix)
Access-Control-Allow-Origin: http://api.weishuo.com
Access-Control-Allow-Methods: GET, POST, PUT
Access-Control-Allow-Headers: X-Custom-Header
Content-Type: text/html; charset=utf-8
Content-Encoding: gzip
Content-Length: 0
Keep-Alive: timeout=2, max=100
Connection: Keep-Alive
Content-Type: text/plain

一旦服务器通过了"预检"请求,以后每次浏览器正常的CORS请求,就都跟简单请求一样,会有一个 Origin 头信息字段。服务器的回应,也都会有一个 Access-Control-Allow-Origin 头信息字段。

SpringBoot设置CORS

目前前后端开发项目,后端主要是使用SpringBoot框架,因此最后再说下SpringBoot如何设置CORS支持跨域。

SpringBoot 支持通过设置 CORS(跨源资源共享)来解决跨域请求问题,核心是使用 @CrossOrigin 注解,有两个地方可以配置该注解,我们选择一种方式即可。

(1)在请求方法上配置

我们可以直接在相应的请求方法上添加 @CrossOrigin 注解,那么该方法则支持跨域。

@RestController
public class WebAPIController {
    @Autowired
    DeviceDataManager deviceDataManager;
 
    @GetMapping("/getDeviceData")
    @CrossOrigin
    public List<DeviceData> getDeviceData() {
        return deviceDataManager. getDeviceData();
    }
}

(2)控制器上配置

我们也可以在控制器上添加 @CrossOrigin 注解,那么该控制器下的所有方法都支持跨域。

@RestController
@CrossOrigin
public class WebAPIController {
    @Autowired
    DeviceDataManager deviceDataManager;
 
    @GetMapping("/getDeviceData")
    public List<DeviceData> getDeviceData() {
        return deviceDataManager. getDeviceData();
    }
}

总结

本文主要介绍了使用CORS来实现跨域。在实际开发过程中,我也经常利用nginx反向代理来解决前后端请求的跨域问题,nginx代理将请求分发到相应项目部署的tomcat服务器上,让前后端项目处于同一个域上,这样也就避免了跨域问题。关于nginx的配置,在后面的文章再做介绍。

【参考资料】

跨域资源共享 CORS 详解。阮一峰。

扫描二维码推送至手机访问。

版权声明:本文由AR编程网发布,如需转载请注明出处。

本文链接:http://www.arlanguage.com/post/486.html

分享给朋友:

“如何解决前后端分离的跨域问题?” 的相关文章

Nginx配置七层负载均衡

Nginx 一般用于七层负载均衡,其吞吐量有一定限制。为了提升系统整体吞吐量,会在 DNS 与 Nginx 之间引入接入层,比如使用LVS(软负载均衡器)、F5(硬负载均衡器)做四层负载均衡。整体的请求流转如下图所示,即首先 DNS 解析到 LVS/F5,然后 LVS/F5 转发给 Nginx,再由...

Python WSGI 高性能服务器 - Gunicorn 简介

一、简介Gunicorn 意即 Green Unicorn,绿色独角兽。它是一个被广泛使用的高性能的 Python WSGI UNIX HTTP服务器,移植自Ruby的独角兽(Unicorn )项目。它具有如下特性:原生支持 WSGI、Django 和 Paster自动工作进程管理简单的 Pytho...

Java 加密解密和数字签名

在做项目中,只要涉及敏感信息,或者对安全有一定要求的场景,都需要对数据进行加密。在Java中原生API即可实现对称加密与非对称加密,并支持常用的加密算法。对称加密对称加密使用单钥完成加解密,加密和解密采用相同的密钥。对称加密的速度快,常用于大量数据进行加密。主流的算法有:AES,3DES。生成3DE...

一文说清nginx规则匹配(含案例分析)

概述: 在nginx中,匹配分很多情形,例如:区分大小写匹配、不区分大小写匹配、有强匹配开头、有模糊匹配某些字符、有匹配后重写动作、有匹配反向代理动作、有匹配后终止操作、还有匹配全局变量类型等等,在很多情形中,匹配动作也存在优先策略,来看看下面的案例。案例一、 server { .............

Linux 系统安装 Nginx

Linux 系统安装 NginxCentOS 7 Nginx 源代码编译安装# 官网下载最新版 https://nginx.org/en/download.html # 系统安装必备组件: yum install gcc gcc-c++ pcre pcre-devel zlib-devel ope...

全网最新最全的Docker命令大集合,收藏备用!

Docker 是一个开源的平台,用于自动化应用程序的部署、扩展和管理。它使得开发者能够打包应用及其依赖项到一个轻量级的容器中,并在任何环境中快速运行。掌握 Docker 的常用命令可以大大提高你的工作效率,尤其是在开发、测试和部署阶段。本文将详细介绍 Docker 的常用命令,涵盖从基本操作到高级使...