开发项目的时候因为浏览器同源策略的限制,经常会遇到跨域问题,本篇文章对常见的跨域解决方案做一个记录。
一,跨域产生的原因
之所以产生跨域主要是因为浏览器同源策略的限制。
同源策略,它是由netspace提出的一个著名的安全策略。
当一个浏览器的两个tab页中分别打开来自百度和谷歌的页面,当浏览器的百度tab页执行一个脚本的时候会检查这个脚本是属于哪个页面的,即检查是否同源,只有和百度同源的脚本才会被执行。如果非同源,那么在请求数据时,浏览器会在控制台中报一个异常,提示拒绝访问。
二,什么情况下算跨域
一个域名地址由以下几个部分组成:
http://www.aaa.com:8080/sie=utf-8&wd=springboot
- 协议:http
- 域名:子域名www,主域名aaa.com
- 端口:8080
从一个域名的网页去请求另一个域名的资源时,协议,域名,端口任意不同,都会出现跨域问题。
http://www.aaa.com:8080——>http://www.aaa.com:8080:同域访问
http://www.aaa.com:8080——>http://www.bbb.com:8080:跨域访问
尤其是在前后端分离的开发模式下,跨域请求是避免不了的。
三,实际演示
下面我们以一个实际功能为例:用户输入用户名密码,发往服务端验证。

因为浏览器同源策略的限制,在浏览器控制台提示我们:
access to xmlhttprequest at ‘http://192.168.1.10:7080/tick-tack/login’ from origin ‘http://192.168.1.10:7060’ has been blocked by cors policy: response to preflight request doesn’t pass access control check: no ‘access-control-allow-origin’ header is present on the requested resource.
我们还可以在network里看到,浏览器在发送我们输入的用户名,密码等数据之前,还发送了一次options的请求,这是浏览器自动发送的,为了验证是否允许跨域访问。

四,解决跨域的方法
我们已经知道,浏览器在发送请求之前会先发送一个options请求,来校验是否允许跨域访问,校验的结果存放在头信息的access-control-allow-origin,因此解决跨域也就是设置头部信息。有四种方法解决跨域。
1,@crossorigin注解
我们可以在特定的某些接口加上@crossorigin注解,表示该接口允许跨域访问。注:未加该注解的接口仍不允许跨域访问
@postmapping("/login")
@crossorigin
public result loginsystem(@requestbody loginuser user) {
if (stringutils.isblank(user.getuseraccount()) || stringutils.isblank(user.getpassword())) {
return result.error(constants.code_400, "参数错误");
}
ticktoken ticktoken = iloginservice.loginsystem(user);
return result.success(ticktoken);
}
@crossorigin注解中的origins还可设置域名,表示只有该域名访问时允许跨域,如:@crossorigin(origins =“http://localhost:7060”);
若origins未设置值,表示所有域名都可以跨域访问该接口

2,添加全局过滤器
若项目中所有接口都允许跨域访问,可增加全局过滤器允许跨域访问。
import org.springframework.context.annotation.bean;
import org.springframework.context.annotation.configuration;
import org.springframework.web.cors.corsconfiguration;
import org.springframework.web.cors.urlbasedcorsconfigurationsource;
import org.springframework.web.filter.corsfilter;
@configuration
public class corsconfig {
// 当前跨域请求最大有效时长。这里默认1天
private static final long max_age = 24 * 60 * 60;
@bean
public corsfilter corsfilter() {
urlbasedcorsconfigurationsource source = new urlbasedcorsconfigurationsource();
corsconfiguration corsconfiguration = new corsconfiguration();
corsconfiguration.addallowedorigin("*"); // 1 设置访问源地址,或者http://localhost:7060
corsconfiguration.addallowedheader("*"); // 2 设置访问源请求头
corsconfiguration.addallowedmethod("*"); // 3 设置访问源请求方法,或设置为"get", "post", "delete", "put"
corsconfiguration.setmaxage(max_age);
source.registercorsconfiguration("/**", corsconfiguration); // 4 对接口配置跨域设置
return new corsfilter(source);
}
}
3,实现webmvcconfigurer
import org.springframework.context.annotation.configuration;
import org.springframework.web.servlet.config.annotation.corsregistry;
import org.springframework.web.servlet.config.annotation.webmvcconfigurer;
@configuration
public class webconfig implements webmvcconfigurer {
// 当前跨域请求最大有效时长。这里默认1天
private static final long max_age = 24 * 60 * 60;
@override
public void addcorsmappings(corsregistry registry) {
registry.addmapping("/**")
.allowedorigins("*")
.allowedmethods("*")
.allowedheaders("*")
.maxage(max_age);
}
}
4,nginx解决跨域
如果项目中有使用nginx来转发请求,那也可以交由nginx来解决跨域,但是有一点需要注意:nginx解决跨域和后端解决跨域最好只保留一个,两种混用会出现很多奇怪的问题。
下面是nginx.conf文件解决跨域的相关配置:
server {
listen 80;
server_name localhost;
location / {
if ($request_method = 'options') {
add_header access-control-allow-origin 'http://localhost:7060';
add_header access-control-allow-headers '*';
add_header access-control-allow-methods '*';
add_header access-control-allow-credentials 'true';
return 204;
}
if ($request_method != 'options') {
add_header access-control-allow-origin 'http://localhost:7060' always;
add_header access-control-allow-credentials 'true';
}
proxy_pass http://localhost:59200;
}
}
5,注意
说明: 文章中很多地方为了方便,access-control-allow-origin设置成了*,这个在开发测试的时候可以这么设置,但如果是生产环境,建议不要设置成*,最好是允许哪些域名访问就设置哪些,毕竟限制域名还是很有必要的。
总结
到此这篇关于spring boot项目中解决跨域问题四种方式的文章就介绍到这了,更多相关springboot解决跨域问题内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论