聊一聊跨域错误CORS

遇到 CORS(跨域资源共享) 错误是前端开发中的"家常便饭"。当浏览器控制台出现类似 Access to XMLHttpRequest from origin 'xxx' has been blocked by CORS policy 的报错时,通常意味着浏览器出于安全考虑,拦截了你的请求。别担心,我为你整理了从"快速解决"到"深度排查"的完整方案。

🛠️ 1. 什么是跨域

在解决之前,请确认是否真的跨域。只要协议、域名、端口三者中任意一个不同,就是跨域。

场景 当前页面 URL 请求接口 URL 是否跨域 原因
1 http://localhost:3000 http://localhost:3000/api 端口、域名、协议完全一致
2 http://localhost:3000 https://localhost:3000/api 协议不同 (http vs https)
3 http://localhost:3000 http://localhost:8080/api 端口不同
4 http://localhost:3000 http://api.example.com 域名不同

🚀 2. 解决方案

根据你的开发环境(前端/后端),选择最适合的一种

💻 方案一:前端开发环境(使用代理 Proxy)

如果你使用的是 ViteWebpack (Vue/React),可以在本地开发服务器配置代理,让请求先发给本地服务器,再由服务器转发给后端。这样浏览器认为是同源请求,不会触发 CORS。

  • Vite 配置 (vite.config.js):

    javascript 复制代码
    export default {
      server: {
        proxy: {
          '/api': {
            target: 'http://localhost:8080', // 后端真实地址
            changeOrigin: true, // 修改请求头中的 origin,让后端以为是它自己发的
            rewrite: (path) => path.replace(/^\/api/, '')
          }
        }
      }
    }
🛡️ 方案二:后端配置 CORS 响应头

最标准解决方式就是后端需要在响应头中明确告诉浏览器:"我允许你这个域名来访问我"。

  • 通用响应头:

    bash 复制代码
    Access-Control-Allow-Origin: http://localhost:3000 
    # 或者 * (但带cookie时不能用*)
    Access-Control-Allow-Methods: GET, POST, PUT, DELETE, OPTIONS
    Access-Control-Allow-Headers: Content-Type, Authorization, X-Requested-With
  • Spring Boot (Java) 示例:

    java 复制代码
    @Configuration
    public class CorsConfig {
        @Bean
        public WebMvcConfigurer corsConfigurer() {
            return new WebMvcConfigurer() {
                @Override
                public void addCorsMappings(CorsRegistry registry) {
                    registry.addMapping("/**")
                            .allowedOriginPatterns("*") // 允许的源
                            .allowedMethods("*")
                            .allowedHeaders("*")
                            .allowCredentials(true); // 允许携带cookie
                }
            };
        }
    }
🗂️ 方案三:Nginx 反向代理

将前端和后端都部署在同一个 Nginx 下,让 Nginx 去做请求转发。对外只有一个域名,从根本上消灭跨域。

配置示例

java 复制代码
server {
    listen 80;
    server_name yourdomain.com;

    # 前端静态资源
    location / {
        root /usr/share/nginx/html;
        try_files $uri $uri/ /index.html;
    }

    # 代理 API 请求到后端
    location /api/ {
        proxy_pass http://localhost:8080; 
        proxy_set_header Host $host;
        proxy_set_header X-Real-IP $remote_addr;
    }
}

🔍 3. 常见报错深度解析

如果你的请求涉及 PUT/DELETE 方法 或者携带了 自定义 Header (如 token),浏览器会先发送一个 OPTIONS 预检请求。常见报错

Response to preflight request doesn't pass access control check

排查步骤

  1. 检查 OPTIONS 请求: 打开浏览器 Network 面板,看 OPTIONS 请求是否返回了 200204
  2. 后端是否处理 OPTIONS: 确保后端路由能正确响应 OPTIONS 请求,并返回正确的 CORS 头(如 Access-Control-Allow-Methods)。
  3. Header 是否匹配: 前端请求头中如果有 X-Token,后端的 Access-Control-Allow-Headers 中必须包含 X-Token

如果你的请求需要携带 Cookie(withCredentials: true),有两点必须注意:

后端 Access-Control-Allow-Origin 不能为 *,必须指定域名(如 http://localhost:3000

前端需设置 credentials: 'include'(Fetch)或 withCredentials = true(XHR)

📌 总结建议

  • 本地开发: 使用 Vite/Webpack 的 Proxy 代理。
  • 联调/测试: 让后端同学开启 CORS 支持。
  • 生产上线: 使用 Nginx 反向代理,既安全又高效。
相关推荐
曲幽1 天前
FastAPI实战:打造本地文生图接口,ollama+diffusers让AI绘画更听话
python·fastapi·web·cors·diffusers·lcm·ollama·dreamshaper8·txt2img
曲幽2 天前
我用FastAPI接ollama大模型,差点被asyncio整崩溃(附对话窗口实战)
python·fastapi·web·async·httpx·asyncio·ollama
曲幽3 天前
FastAPI + Ollama 实战:搭一个能查天气的AI助手
python·ai·lora·torch·fastapi·web·model·ollama·weatherapi
holeer5 天前
【V1.0】Typora 中的 HTML 支持|软件文档自翻译
前端·编辑器·html·typora·web·markdown·文档
敲敲了个代码6 天前
浏览器时间管理大师:深度拆解 5 大核心调度 API
前端·javascript·学习·web
差不多的张三7 天前
【RSS】博客收集
web
夜瞬8 天前
【Flask 框架学习】02:核心基本概念全解析
python·flask·web
kyle~8 天前
Python---webbrowser库 跨平台打开浏览器的控制接口
开发语言·python·web
Rhystt9 天前
furryCTF题解|Web方向|ezmd5、猫猫最后的复仇
android·前端·web安全·web
REDcker12 天前
Media Source Extensions (MSE) 详解
前端·网络·chrome·浏览器·web·js