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

Spring Boot跨域问题终极解决方案:3种方案彻底告别CORS错误

ruisui883周前 (04-07)技术分析18

引言

"接口调不通?前端同事又双叒叕在吼跨域了!"
"明明Postman能通,浏览器却报OPTIONS 403?"
"生产环境跨域配置突然失效,凌晨3点被夺命连环Call?"

每个后端开发者都经历过被CORS(跨域资源共享)支配的恐惧。本文将手把手教你3种Spring Boot跨域解决方案,从注解到源码配置,从单接口到全局管控,附带防踩坑指南线上应急预案,从此告别跨域噩梦!


一、跨域问题本质:30秒搞懂CORS核心机制

  1. 浏览器安全策略:同源策略(Same-Origin Policy)限制不同源的前后端交互
  2. CORS破局关键:服务端通过HTTP响应头声明允许的跨域规则
  3. 核心响应头

http

Access-Control-Allow-Origin: *                // 允许的域名(*为通配符)  
Access-Control-Allow-Methods: GET,POST        // 允许的HTTP方法  
Access-Control-Allow-Headers: Content-Type    // 允许的请求头  
Access-Control-Max-Age: 3600                  // 预检请求缓存时间(秒)  

二、Spring Boot解决跨域的3种实战方案

方案1:@CrossOrigin注解(精准打击)

适用场景:仅需开放特定接口的跨域访问

java

@RestController
public class UserController {
    
    // 单个方法配置
    @CrossOrigin(origins = "https://your-frontend.com", 
                 methods = {RequestMethod.GET, RequestMethod.POST},
                 allowedHeaders = "Content-Type")
    @GetMapping("/api/user")
    public User getUser() {
        return new User("码农", 28);
    }

    // 类级别配置(作用于所有接口)
    @CrossOrigin(origins = "*")
    @RestController
    public class OpenApiController {
        // ...
    }
}

坑点预警

  • 若同时存在全局配置,注解配置会被覆盖!
  • 不支持通配符子域名(如 *.domain.com)

方案2:全局配置(一劳永逸)

推荐指数:★★★★★
通过实现WebMvcConfigurer统一管理:

java

@Configuration
public class CorsConfig implements WebMvcConfigurer {

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**")              // 所有接口
                .allowedOrigins("https://prod.com", "http://localhost:8080") 
                .allowedMethods("*")             // 允许所有方法
                .allowedHeaders("*")             // 允许所有头
                .allowCredentials(true)         // 允许携带Cookie
                .maxAge(1800);                   // 预检请求缓存时间
    }
}

避坑指南

  • allowCredentials(true)时,allowedOrigins不能为*,必须明确指定域名!
  • 若使用Spring Security,需额外配置cors()+csrf().disable()(见方案3)

方案3:过滤器(终极自由)

适用场景:需要动态控制跨域规则(如多租户系统)

java

@Bean
public FilterRegistrationBean corsFilter() {
    UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
    CorsConfiguration config = new CorsConfiguration();
    
    config.setAllowCredentials(true);
    config.addAllowedOrigin("https://*.company.com"); // 支持通配符子域名
    config.addAllowedMethod("*");
    config.addAllowedHeader("*");
    config.setMaxAge(3600L);

    source.registerCorsConfiguration("/**", config);
    return new FilterRegistrationBean<>(new CorsFilter(source));
}

优势

  • 优先级最高,可覆盖其他配置
  • 支持复杂逻辑(如从数据库读取允许的域名)

三、深度避坑:那些官方文档不会告诉你的秘密

1. 预检请求(Preflight)的阴谋

  • 现象:前端POST请求变成OPTIONS请求
  • 本质:浏览器对非简单请求自动发起预检请求
  • 解决方案
    • 确保服务器正确处理OPTIONS方法(Spring Boot默认支持)
    • 设置合理的maxAge减少预检次数

2. Spring Security的偷袭

症状:明明配置了跨域,却依然返回403
修复方案

java

@Configuration
@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {
    
    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http.cors()                          // 启用CORS支持
            .and()
            .csrf().disable()               // 禁用CSRF(根据业务需要)
            .authorizeRequests()
            .anyRequest().permitAll();
    }

    // 关键!提供CorsConfigurationSource
    @Bean
    public CorsConfigurationSource corsConfigurationSource() {
        CorsConfiguration configuration = new CorsConfiguration();
        configuration.applyPermitDefaultValues();
        configuration.addAllowedMethod(HttpMethod.PUT); // 按需添加
        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
        source.registerCorsConfiguration("/**", configuration);
        return source;
    }
}

3. 生产环境突发跨域故障应急方案

  1. 临时允许所有域名(慎用!):
  2. java
allowedOrigins("*")
  1. Nginx层统一加响应头
  2. nginx
location / {
    add_header 'Access-Control-Allow-Origin' '*';
    add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS';
    add_header 'Access-Control-Allow-Headers' 'DNT,User-Agent,X-Requested-With,Content-Type';
}
  1. 接口代理:通过后端调用第三方接口,规避浏览器限制

四、最佳实践总结

方案

适用场景

优点

缺点

@CrossOrigin

快速验证、少量接口

简单易用

配置分散,维护成本高

全局配置

标准项目

统一管理,优先级适中

无法动态调整

过滤器

需要动态规则/微服务网关

灵活性强,优先级最高

实现稍复杂


结语

跨域不是技术难题,而是前后端协作的试金石。掌握这3种方案后,下次再遇到跨域问题,你可以淡定地说:“给我1分钟,马上搞定!”

转发收藏本文,让你团队的前端小伙伴不再抓狂!

讨论话题:你在解决跨域问题时踩过哪些坑? 评论区分享经验,点赞最高的送《Spring Boot实战派》电子书!


附录:常用检测工具

  1. Chrome开发者工具 → Network标签查看请求头/响应头
  2. Postman → 关闭CORS限制验证接口本身
  3. 在线CORS检测工具:https://test-cors.org/

关于作者
资深全栈开发者,曾因CORS问题被前端追杀三条街。专注分享实战开发技巧,关注我,解锁更多《Spring Boot避坑指南》!

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

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

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

标签: 同源策略
分享给朋友:

“Spring Boot跨域问题终极解决方案:3种方案彻底告别CORS错误” 的相关文章

一套智能停车场收费管理系统设计方案,拓扑图VISIO格式

大家好,我是薛哥。最近VIP会员群的读者咨询停车场管理系统的规划设计方案,今天分享一个模板素材,主要里面的拓扑图可以编辑的,VISIO格式,建议收藏备用。此套完整的Word方案,VIP会员下载!智能停车场收费管理子系统1、系统概述本次停车场管理系统设计纯车牌识别系统,并可在合适的位置设置中央收费点,...

如何在 Linux 发行版中安装微信和 QQ?

很多人因为工作沟通的原因需要用到微信和 QQ,那么如何在 Linux 发行版中安装微信和 QQ 呢?以下是一些尝试的解决方法。QQ上一个版本的 QQ Linux 版还是在2009年,而在现在,基于 NT 架构的全新 QQ Linux版已经被正式推出,为所有用户提供下载。新版本提供了deb、rpm、A...

java调用API操作GitLab

最近需要在一个WEB项目中集成GitLab,用到了GitLab的API操作,在网上找了很久都是说直接调用GitLab的Http接口,而且API官方只有javadoc没有其它说明文档,特别记录下,以备查询。这里采用Token的认证方式,因此需要先登陆GitLab新建一个Token,创建方式如下:创建完...

一次Java内存占用高的排查案例,解释了我对内存问题的所有疑问

问题现象7月25号,我们一服务的内存占用较高,约13G,容器总内存16G,占用约85%,触发了内存报警(阈值85%),而我们是按容器内存60%(9.6G)的比例配置的JVM堆内存。看了下其它服务,同样的堆内存配置,它们内存占用约70%~79%,此服务比其它服务内存占用稍大。那为什么此服务内存占用稍大...

我的VIM配置

写一篇关于VIM配置的文章,记录下自己的VIM配置,力求简洁实用。VIM的配置保存在文件~/.vimrc中(Windows下是C:\Users\yourname \_vimrc)。VIM除了自身可配置项外,还可插件扩展。VIM的插件一般用vundle或vim-plug来管理,但我力求简单,不打算装太...

有效地简化导航-Part 1:信息架构

「四步走」——理想的导航系统要做一个可用的导航系统,网页设计师必须按顺序回答以下4个问题:1. 如何组织内容?2. 如何解释导航的选项?3. 哪种导航菜单最适合容纳这些选项?4. 如何设计导航菜单?前两个问题关注构建和便签内容,通常称为信息架构。信息架构师通常用网站地图(site map diagr...