什么是跨域访问问题,如何解决?

跨域访问问题(Cross-Origin Resource Sharing, CORS)是由浏览器的同源策略(Same-Origin Policy)引发的一种安全限制。当网页尝试从一个与自身"源"(协议、域名、端口号)不同的地址请求资源(如API数据、图片、脚本等)时,浏览器会阻止该请求,除非目标服务器明确允许这种跨域访问。


核心概念解析

  1. 同源策略(Same-Origin Policy)

    • 定义:浏览器要求网页只能访问与自身"源"相同的资源。
    • 同源判定 :协议(http/https)、域名(example.com)、端口(:80)三者完全一致。
      例如:
      https://a.com/index.htmlhttps://a.com/api同源
      https://a.com/index.htmlhttp://a.com/api ❌ 协议不同(HTTPS vs HTTP)
      https://a.com/index.htmlhttps://b.com/api ❌ 域名不同
      https://a.com:80/index.htmlhttps://a.com:8080/api ❌ 端口不同
  2. 跨域场景举例

    • 前端部署在 https://web.com,调用API https://api.com/data
    • 本地开发时 http://localhost:3000 访问本地API http://localhost:8000(端口不同)。
    • 子域名不同:https://shop.example.com 访问 https://api.example.com

为什么需要跨域限制?

  1. 安全防护
    防止恶意网站通过脚本窃取用户敏感数据(如Cookie、私有数据)。例如:
    • 用户登录了 bank.com,若恶意网站 evil.com 能随意读取 bank.com 的数据,将导致账户被盗。
  2. 隔离攻击面
    限制CSRF(跨站请求伪造)、XSS(跨站脚本攻击)等漏洞的影响范围。

如何解决跨域问题?

方案1:服务端配置CORS(主流方案)

服务端在响应头中添加允许跨域的声明:

http 复制代码
Access-Control-Allow-Origin: https://web.com   // 允许特定域名
Access-Control-Allow-Origin: *                 // 允许所有域名(慎用)
Access-Control-Allow-Methods: GET, POST, PUT   // 允许的请求方法
Access-Control-Allow-Headers: Content-Type     // 允许的请求头

流程

  1. 浏览器发送预检请求(OPTIONS)询问服务器是否允许跨域。
  2. 服务器响应允许的规则。
  3. 浏览器确认后发送真实请求。
方案2:代理服务器(Proxy)

前端通过同源代理中转请求:

复制代码
浏览器 → https://web.com/proxy?target_api=xxx → 代理请求 → https://api.com/data

常用于开发环境(如Webpack DevServer代理)或后端服务中转。

方案3:JSONP(过时方案,仅限GET请求)

利用 <script> 标签不受同源策略限制的特性:

html 复制代码
<script src="https://api.com/data?callback=handleData"></script>

服务端返回:handleData({...}),前端需提前定义 handleData 函数。


常见误区

  • "Postman能请求成功,但浏览器报CORS错误"
    → 因为Postman不受同源策略限制,浏览器才会拦截。
  • "前端代码修改请求头可以绕过CORS"
    → 浏览器会阻止前端修改敏感请求头(如Origin),必须服务端授权。

总结

关键点 说明
触发原因 浏览器同源策略的安全限制
本质 浏览器与服务端的协作机制(通过HTTP头协商)
解决方案 服务端配置CORS、代理服务器、JSONP(历史方案)
核心目标 在安全前提下实现合法跨域通信

⚠️ 注意:CORS是浏览器行为,服务端即使未配置CORS,接口本身仍可被非浏览器工具(如curl、Postman)正常调用。

相关推荐
skeletron20113 分钟前
🚀AI评测这么玩(2)——使用开源评测引擎eval-engine实现问答相似度评估
前端·后端
前端开发爱好者3 分钟前
Vite 7.1.1 疑似遭受大规模 "攻击"!
前端·vue.js·vite
小徐_23335 分钟前
uni-app 还在手写请求?alova 帮你全搞定!
前端·uni-app·axios
shark_chili14 分钟前
颠覆认知!这才是synchronized最硬核的打开方式
后端
七十二時_阿川17 分钟前
React 浏览器重新绘制之前副作用之useLayoutEffect
前端·程序员
就是帅我不改17 分钟前
99%的Java程序员都写错了!高并发下你的Service层正在拖垮整个系统!
后端·架构
Apifox18 分钟前
API 文档中有多种参数结构怎么办?Apifox 里用 oneOf/anyOf/allOf 这样写
前端·后端·测试
似水流年流不尽思念18 分钟前
如何实现一个线程安全的单例模式?
后端·面试
楽码21 分钟前
了解HMAC及实现步骤
后端·算法·微服务
IT小番茄23 分钟前
若依框架导出 Excel 签名实现方案
前端·后端