当前位置: 首页 > news >正文

跨域问题解决方案汇总

全文默认讲的是浏览器端发起的 HTTP 请求的“跨域”问题(同源策略导致的受限)。

跨域 / 同源策略概述

  • ​**同源(same-origin)**​:协议、域名(host)、端口 三者完全相同称为同源。 例如 https://example.com:443http://example.com 不是同源(协议不同)。
  • ​**同源策略(SOP)**​:浏览器的一种安全机制,限制从一个源加载的脚本去读取另一个源的响应(以防 CSRF / 数据泄露)。
  • ​**跨域(cross-origin)**​:当请求目标不满足同源时即为跨域,请求仍可发出,但浏览器会阻止 JS 读取响应,除非服务器明确允许(即 CORS -> 跨域资源共享)。

CORS(Cross-Origin Resource Sharing)

CORS(Cross-Origin Resource Sharing)跨域资源共享。浏览器会根据响应头判断是否允许跨域读取。一些关键的响应头包括:

  • Access-Control-Allow-Origin:允许的源(或 *)。

    但是这里我们一般不配置为 *,因为如果响应包含敏感数据或依赖 cookie/凭证(Authorization / session), *Access-Control-Allow-Credentials: true 不能同用,浏览器也会拒绝这种组合,属于安全漏洞 ⚠️。(篇幅有限,更多细节见下篇文章。)

  • Access-Control-Allow-Methods:允许的方法(GET, POST, PUT…)。
  • Access-Control-Allow-Headers:允许的自定义 Header(如 Authorization, X-Custom-Header)。
  • Access-Control-Allow-Credentials:是否允许带 cookie/凭证(true 表示允许)。
  • Access-Control-Expose-Headers:允许前端访问的响应头。
  • Access-Control-Max-Age:预检(preflight)结果缓存时长(秒)。

    预检请求(preflight):当请求使用了非“简单请求”方法或自定义了 header、或 Content-Type 非简单类型时,浏览器会先发 OPTIONS 请求询问服务器是否允许。

常见解决方案

方案 A — 在服务端正确配置 CORS

这是最通用也最推荐的做法:​在响应里返回正确的 CORS 头​。

Express(Node)示例(使用 cors 中间件)
const express = require('express')
const cors = require('cors')
const app = express()app.use(cors({origin: 'https://app.example.com', // 注意不能用 '*' 配合 credentialsmethods: ['GET','POST','PUT','DELETE','OPTIONS'],credentials: true, // 允许 cookieallowedHeaders: ['Content-Type','Authorization','X-Requested-With']
}));
Nginx:把 CORS header 加到响应上
location /api/ {if ($request_method = 'OPTIONS') {add_header 'Access-Control-Allow-Origin' 'https://app.example.com';add_header 'Access-Control-Allow-Methods' 'GET,POST,OPTIONS,PUT,DELETE';add_header 'Access-Control-Allow-Headers' 'Authorization,Content-Type';add_header 'Access-Control-Allow-Credentials' 'true';add_header 'Access-Control-Max-Age' 1728000;add_header 'Content-Length' 0;add_header 'Content-Type' 'text/plain charset=UTF-8';return 204;}proxy_pass http://backend;proxy_set_header Host $host;proxy_set_header X-Real-IP $remote_addr;# 把 header 添加到后端响应proxy_hide_header 'Access-Control-Allow-Origin';add_header 'Access-Control-Allow-Origin' 'https://app.example.com';add_header 'Access-Control-Allow-Credentials' 'true';
}

方案 B — 前端代理到同源(仅开发阶段)

开发阶段或没有后端权限时使用我们经常使用构建工具的 dev server 实现反向代理。把跨域请求代理到本地 dev server(同源),由 dev server 转发到目标服务器。

Vite dev server 配置
// vite.config.js
export default {server: {proxy: {'/api': {target: 'https://api.example.com',changeOrigin: true,rewrite: path => path.replace(/^\/api/, '')}}}
}

方案 C — Nginx 反向代理(生产阶段常用)

这也是生产环境下的常用方案。在 Nginx 等反向代理层统一转发,前端调用同域(Nginx),Nginx 代理后再与后端跨域通信。

前端请求:https://app.example.com/api/...(同源) Nginx proxy_pass -> https://api.example.com/...

server {listen 80;server_name example.com;# 静态资源代理location /static/ {root /project/static;index index.html index.html; # 访问 /static/ 会自动“重定向”到 /project/static/index.html 文件}# API代理location /api/ {rewrite ^/api/(.*)$ /$1 break; # 如果服务器没有 /api 记得重写proxy_pass http://api.example.com; # 代理的地址}
}

方案 D — JSONP(已过时,仅限 GET)

只支持 GET,通过 <script> 标签绕过 SOP,服务端返回 callback(...),容易被攻击者使用 callback 恶意函数做 XSS 攻击。

方案 E — postMessage(跨窗口/iframe 场景)

当需要跨域页面间通信(iframe 和父窗口),使用 window.postMessage。适用于页面间数据交换,不适用于普通 API 请求。我们一般会在微前端、OAuth 第三方登录、单点登录、支付页面回调、WebView 混合开发中使用。

方案 F — WebSocket(不受 CORS 限制)

WebSocket 握手不是标准的 CORS;如果用 WS/WSS,浏览器不会因同源限制阻止读取消息(但服务器可能做 origin 校验)。我们也不可能为了跨域强行使用 WebSocket。(之后会详细介绍 HTTP 协议和 WebSocket 协议关系和他们的使用)

开发时一些坑

在我刚开始独立从零到一搭建前后端项目的时候(当时还没有什么 AI Coding,全凭一手搜索引擎),这个问题让我红温到晚上一点也没有解决(没错,当时我还很菜 hh)。

处理带 cookie / 带凭证的跨域请求

  1. 服务端:
    1. Access-Control-Allow-Origin: https://app.example.com(具体 origin,不能是 *
    2. Access-Control-Allow-Credentials: true
  2. 前端(fetch / xhr):
    1. fetch(url, { credentials: 'include' })xhr.withCredentials = true

当时我犯的错误​:

  • 没有设置 credentials: 'include',浏览器不会发送 cookie。
  • 服务端回送 Access-Control-Allow-Origin: *Allow-Credentials: true 同时存在(浏览器会拒绝)。
http://www.dtcms.com/a/614119.html

相关文章:

  • 30-ESP32-S3开发
  • C语言编译爱心 | 深入浅出解析C语言编译过程及技巧
  • SPRING_CACHE_REDIS_技术总结
  • 【LeetCode热题100(67/100)】有效的括号
  • 网站建设公司彩铃织梦后台做的网站怎么绑定域名
  • 汇编语言编译器的作用 | 理解汇编编译过程与程序优化技巧
  • 网站备案每年一次吗网站建设实训的方法
  • 怎样查看网站的权重做微信网站公司哪家好
  • Typora picgo-core gitee图片上传设置
  • 哪个网站发布招聘信息免费淮安网站建设公司电话
  • 企业网站优化做法模拟网站效果
  • 第29集科立分板机:降本增效新选择科立自动化分板机赋能电子企业高质量发展
  • Revit 200+新功能之“构件赋房间属性”,解决特殊族提取房间错乱问题!中心点不落在房间内,进而无法正确识别所在房间!
  • 编辑器和笔记软件汇总(三):NotebookLM、note-gen、MiaoYan、LetsMarkdown、DocFlow
  • openGauss 企业级开源数据库架构深度解析
  • 伸展树分析
  • 通州做网站公司市场营销策划包括哪些内容
  • 嵌入式开发学习日志43——FreeRTOS之引入
  • 阳信住房和城乡建设厅网站小城镇建设的网站文献
  • 3542. 将所有元素变为 0 的最少操作次数
  • 宁波商城网站建设互联网创意网站有哪些方面
  • 基于Vue的售票系统开发3g480(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
  • 06-文件操作-练习
  • 中企动力做网站要全款全能网站模板
  • 花瓣设计网站官网入口广州分公司注册
  • QPushButton弹出菜单的完整实现指南
  • 【项目】个人博客测试报告
  • Linux下通过sysfs读写GPIO的一个注意事项
  • Metasploitable2靶场全部漏洞超详细讲解(含Metasploitable2靶场下载)
  • 复变函数与积分变换 第三章——复变函数的积分