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

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

ruisui884个月前 (03-07)技术分析16

跨域问题是由于浏览器的一种安全策略而出现的,即同源策略(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以外,还有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请求过程如上图所示,在正式通信之前,会增加一次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,表示请求来自哪个源。

服务器收到"预检"请求以后,检查了Origin
Access-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 getDeviceData() {
        return deviceDataManager. getDeviceData();
    }
}

(2)控制器上配置

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

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

总结

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

【参考资料】

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

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

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

本文链接:http://www.ruisui88.com/post/2575.html

分享给朋友:

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

程序员开发必会之git常用命令,git配置、拉取、提交、分支管理

整理日常开发过程中经常使用的git命令![送心]git配置SSH刚进入项目开发中,我们首先需要配置git的config、配置SSH方式拉取代码,以后就免输入账号密码了!# 按顺序执行 git config --global user.name "自己的账号" git config -...

面试被逼疯:聊聊Python Import System?

面试官一个小时逼疯面试者:聊聊Python Import System?对于每一位Python开发者来说,import这个关键字是再熟悉不过了,无论是我们引用官方库还是三方库,都可以通过import xxx的形式来导入。可能很多人认为这只是Python的一个最基础的常识之一,似乎没有可以扩展的点了,...

el-table内容\n换行解决办法

问题请求到的数据带有换行符 '\n'但页面展示时不换行statusRemark: "\"1、按期完成计划且准确率100%,得100分;\n2、各项目每延误1天,扣1分;每失误1次或者员工投诉1次,扣3分,失误层面达到公司级影响较大的,该项绩效分数为0\"\n&...

双子座应用程序推出模型切换器以在Android上访问2.0

#头条精品计划# 快速导读谷歌推出了Gemini 2.0 Flash实验版,现已在其安卓应用中可用,之前仅在gemini.google.com网站上提供。新版本的15.50包含模型切换器,用户可以在设置中选择不同模型,包括1.5 Pro、1.5 Flash和2.0 Flash实验版。谷歌提醒,2.0...

佳能 EOS R8 深度评测

佳能 EOS R8 的定位是入门级全画幅无反光镜可换镜头相机。尽管在产品阵容中处于这一位置,R8 仍然是一个强大的相机,配备了先进的 R6 II 同款成像传感器、快速处理器和令人难以置信的自动对焦系统,体积小、重量轻、价格低。这款相机是发烧友、旅行者、家庭以及任何想要全画幅传感器相机的人的绝佳选择。...

Python中的11 种数组算法

1. 创建数组 创建数组意味着留出一个连续的内存块来存储相同类型的元素。在大多数语言中,您可以在创建数组时指定数组的大小。假设您正在书架上整理一组书籍,并且您需要为正好 10 本书预留空间。功能架上的每个空间都对应于数组中的一个索引。# Example in Python arr = [1, 2,...