[ 跨域问题 ] 前后端以及服务端 解决跨域的各种方法

这篇文章主要介绍了跨域问题,包括其定义、产生原因及各种解决方法。原因是浏览器安全策略限制,方法有 JSONP、CORS、Domain、 postMessage、Nginx配置、.NetCore配置。

 前言

什么是跨域问题?

在Web应用中,当一个网页的脚本试图去请求另一个域名下的资源时,就会遇到跨域问题。跨域问题是由浏览器的同源策略所引起的。换句话说:后端返回给浏览器的数据会被浏览器的同源策略给拦截下来。

同源策略要求资源的协议、域名和端口号都必须相同,才能确保数据的安全性。如果不满足这个条件,请求将被浏览器拒绝,从而导致跨域问题的出现。

  • 同源策略: 协议号-域名-端口号 都相同的地址,浏览器才认为是同源
协议号:域名:端口号 / 路径https://192.168.31.45:8080/userhttps://192.168.31.45:8080/list

上面这个例子虽然它们的路径不一样但是协议号、域名、端口号都相同,所以它们就是同源的

跨域问题的原因?

跨域问题主要是由于浏览器的安全策略限制引起的。同源策略的目的是保护用户的隐私和数据安全,防止恶意网站获取用户的敏感信息或进行未授权的操作。 通过限制跨域请求,浏览器有效地减少了许多网络攻击的风险,例如跨站脚本攻击(XSS)和跨站请求伪造(CSRF)。

JSONP

JSONP(JSON with Padding)是一种用于解决跨域请求的技术,它利用了 <script> 标签可以跨域加载资源的特性。

下面是 JSONP 解决跨域请求的基本原理:

  1. 前端发起 JSONP 请求: 前端页面通过动态创建 <script> 标签,设置其 src 属性为包含回调函数的 URL。通常这个 URL 是指向另一个域名下的服务器接口。

  2. 服务端返回数据: 服务端接收到 JSONP 请求后,会将数据包装在回调函数中返回给前端。这样前端页面就可以获得跨域请求返回的数据。

  3. 前端处理数据: 前端页面定义好与回调函数同名的 JavaScript 函数,当服务端返回数据时,会执行这个函数并传入返回的数据作为参数,从而实现跨域数据的获取和处理。

前端代码:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><button id="btn">获取数据</button><script>// 定义一个函数 jsonp,用于发送 JSONP 请求function jsonp(url, cb) {return new Promise((resolve, reject) => {// 创建一个 script 标签const script = document.createElement('script');// 设置 script 的 src 属性,包含了请求的 URL 和回调函数名称script.src = `${url}?cb=${cb}`; // http://localhost:3000?cb='callback'// 将 script 添加到文档中document.body.appendChild(script); // 浏览器自动请求 src 中的内容// 定义一个全局函数,用于处理返回的数据window[cb] = (data) => {resolve(data)}})}// 获取按钮元素let btn = document.getElementById('btn');// 绑定点击事件btn.addEventListener('click', () => {// 发送 JSONP 请求jsonp('http://localhost:3000', 'callback').then(res => {console.log('后端的返回结果:' + res);})})</script>
</body>
</html>

后端代码:

const Koa = require('koa');
const app = new Koa();// 定义中间件函数 main,处理请求并返回数据
const main = (ctx, next) => {console.log(ctx.query); // 输出请求参数对象 { cb: 'callback' }// 从请求参数中获取回调函数名称const cb = ctx.query.cb;// 准备要返回给前端的数据const data = '给前端的数据';// 构造带有回调函数名称的字符串,格式为 'callback("给前端的数据")'const str = `${cb}('${data}')`;// 将构造好的字符串作为响应体返回给前端ctx.body = str;
}// 将 main 中间件注册到 Koa 应用中
app.use(main);// 监听 3000 端口,启动服务器
app.listen(3000, () => {console.log('listening on port 3000');
})

优点:

  1. 简单易用: JSONP 实现简单,只需在前端添加一个 <script> 标签即可完成跨域请求,无需复杂的配置。
  2. 兼容性好: JSONP 能够兼容各种浏览器,包括早期版本的浏览器,因为它是通过动态创建 <script> 标签实现的。
  3. 支持跨域请求: JSONP 可以在不同域之间进行数据通信,解决了传统 AJAX 请求受同源策略限制的问题。

缺点:

  1. 安全性问题: JSONP 存在安全风险,因为它是通过在前端动态加载脚本来获取数据,可能会被用于注入恶意脚本,导致安全漏洞。
  2. 只支持 GET 请求: JSONP 只能发起 GET 请求,无法支持其他类型的 HTTP 请求,如 POST、PUT 等。
  3. 依赖服务端支持: JSONP 需要服务端返回数据时将其包裹在一个回调函数中,因此需要服务端提供支持,如果服务端不支持 JSONP 格式返回数据,则无法使用该方法。

CORS(跨域资源共享)

CORS是一种机制,允许服务器在响应中携带一个特殊的标头,以告知浏览器该服务器允许哪些源的网页访问其资源。 可以总结为一句话:后端通过设置响应头来告诉浏览器不要拒绝接受后端的响应。

前端代码:在用户点击按钮时,通过发送跨域请求获取服务器返回的数据,并将数据打印到浏览器的控制台

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><button id="btn">获取数据</button><script>let btn = document.getElementById('btn');btn.addEventListener('click', () => {fetch('http://localhost:3000')  // 发送跨域请求到服务器.then(res => res.json()).then((res) => {console.log(res);  // 打印返回的数据到控制台})})</script>
</body>
</html>

代码实现了以下功能:

  1. 在 HTML 中定义了一个按钮元素,id 为 "btn",用于触发发送数据请求。
  2. 在 JavaScript 部分,使用 fetch 函数发送 GET 请求到指定的服务器地址 "http://localhost:3000"。
  3. 在成功接收到服务器响应后,使用 .then 方法将响应解析为 JSON 格式。
  4. 在第二个 .then 方法中,处理解析后的数据,并将其输出到浏览器的控制台。

后端代码 :Node.js 服务器代码,创建了一个 HTTP 服务器,监听在端口 3000。当接收到请求时,返回一个包含 "hello cors" 消息的 JSON 数据。

const http = require('http');const server = http.createServer((req, res) => {res.writeHead(200, {// cros实现原理'Access-Control-Allow-Origin': 'http://127.0.0.1:5500'  // 允许来自指定地址的跨域请求})let data = {msg: "hello cors"}res.end(JSON.stringify(data)) // 返回数据给前端页面
})server.listen(3000, () => {console.log('listening on port 3000');
})

后端代码第6行在服务端设置响应头来控制跨域访问

常见的响应头包括:

  • Access-Control-Allow-Origin:指定允许访问的域名。

  • 例如,Access-Control-Allow-Origin: *表示允许所有域名访问,

  • Access-Control-Allow-Origin:'http://127.0.0.1:5500'表示只允许特定域名访问。

  • Access-Control-Allow-Methods:指定允许的HTTP方法。

  • Access-Control-Allow-Headers:指定允许的自定义请求头。

CORS支持各种类型的HTTP请求,包括GET、POST等。

Domain

我们还可以使用 Domain 方法来解决一些特定情况下的跨域访问问题。 在跨域通信时,还需要注意以下几点:

  1. 页面的域名必须满足 Domain 方法的限制,即二级域名相同(如 example.com)。

  2. 父级页面需要在设置 document.domain 之前定义需要共享的变量或对象。

  3. 子级页面可以通过 window.parent 来访问父级页面的属性或变量,但需要确保父级页面已经加载完成并且两者的域名设置已生效。

下面举一个简单的例子来说明 Domain 方法的用法:

假设有两个页面分别位于不同子域名下,一个是 parent.example.com,另一个是 child.example.com。我们希望这两个页面能够进行跨域通信。

在父级页面 parent.example.com/index.html 中的代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Parent Page</title>
</head>
<body><h1>Parent Page</h1><iframe src="http://child.example.com/child.html"></iframe><script>document.domain = 'example.com';var messageFromParent = 'Hello from parent page!';</script>
</body>
</html>

在子级页面 child.example.com/child.html 中的代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Child Page</title>
</head>
<body><h1>Child Page</h1><script>document.domain = 'example.com';var messageFromChild = 'Hello from child page!';// 访问父级页面定义的变量var message = window.parent.messageFromParent;console.log('Message from parent: ' + message);</script>
</body>
</html>

在这个例子中,父级页面和子级页面分别设置了相同的 document.domain 为 'example.com',以实现二级域名相同。

父级页面定义了一个名为 messageFromParent 的变量,子级页面在加载后通过 window.parent 来访问父级页面定义的变量,并打印出父级页面的消息。

通过设置相同的 document.domain,父子页面之间就可以进行跨域通信,实现数据共享和交互。

postMessage

使用 postMessage() 方法结合 <iframe> 元素可以实现跨域通信,这是一种常见的技术。通过在父窗口和嵌套的 <iframe> 之间使用 postMessage() 方法,可以安全地在不同源之间进行通信。

<iframe> 可以用于解决跨域通信的问题,其原理是利用浏览器中同源策略的限制,将不同域的内容加载到独立的 <iframe> 中,通过 postMessage 方法进行跨文档通信。

在 a.html 文件中:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><h2>a.html</h2><!-- 创建一个 iframe 元素来加载 b.html --><iframe src="http://127.0.0.1:5500/postMessage/b.html" frameborder="0" id="iframe"></iframe><script>// 向 b.html 发送数据let iframe = document.getElementById('iframe');iframe.onload = function() {// 准备要发送的数据let data = {name: 'Tom'};// 通过 postMessage 方法向 iframe 发送数据iframe.contentWindow.postMessage(JSON.stringify(data), 'http://127.0.0.1:5500');}// 监听来自 b.html 的消息window.addEventListener('message', function(e) {console.log(e.data);});</script>
</body>
</html>

在 b.html 文件中:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><h4>b.html</h4><script>// 监听来自父页面的消息window.addEventListener('message', function(e) {console.log(JSON.parse(e.data));if (e.data) {// 收到消息后延迟 2 秒发送回应给父页面setTimeout(function() {window.parent.postMessage('我接受到', 'http://127.0.0.1:5500');}, 2000);}});</script>
</body>
</html>

优点:

  • 安全性:  使用 <iframe> 结合 postMessage 进行跨域通信是一种相对安全的方法,能够避免常见的跨站脚本攻击(XSS)。
  • 灵活性:  可以在不同域之间传递数据,实现更丰富的交互体验,如单点登录、共享数据等。
  • 适用性广泛:  跨域通信是 Web 开发中常见需求,而 <iframe> 结合 postMessage 是一种通用且有效的解决方案。

缺点:

  • 复杂性:  跨域通信涉及到多个文档之间的交互,需要额外的处理和编码,增加了开发的复杂度。
  • 性能开销:  使用 <iframe> 进行跨域通信可能会引入额外的网络请求和资源加载,对页面加载性能有一定影响。
  • 兼容性:  旧版本的浏览器可能对 postMessage 支持不完整,需要做兼容性处理。

Nginx 

nginx反向代理 配置 

这个配置允许任何域通过GET、POST和OPTIONS方法访问资源,并且允许一些常见的头信息字段。Access-Control-Max-Age 指令用于指定预检请求的结果能被缓存多久。

确保在实际部署时,根据安全和需求情况将Access-Control-Allow-Origin设置为特定域,而不是*(表示允许所有域),以减少跨站脚本攻击(XSS)的风险。

server {listen 80;server_name example.com;location / {# 设置允许跨域的域,* 表示允许任何域,也可以设置特定的域add_header 'Access-Control-Allow-Origin' '*';# 允许的方法add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS';# 允许的头信息字段add_header 'Access-Control-Allow-Headers' 'User-Agent,Keep-Alive,Content-Type';# 缓存时间add_header 'Access-Control-Max-Age' 1728000;# 其他配置...}# 其他 server 配置...
}

.NetCore

在.NET Core中配置跨域非常简单。你可以在Startup.cs文件中的ConfigureServices方法添加跨域服务,并在Configure方法中配置跨域。

public class Startup
{// ...// 在ConfigureServices方法中添加跨域服务public void ConfigureServices(IServiceCollection services){// ...services.AddCors(options =>{options.AddPolicy("CorsPolicy",builder => builder.AllowAnyOrigin().AllowAnyMethod().AllowAnyHeader());});}// 在Configure方法中配置跨域public void Configure(IApplicationBuilder app, IWebHostEnvironment env){// ...// 使用跨域策略app.UseCors("CorsPolicy");// ...}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/60206.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

论文阅读——Intrusion detection systems using longshort‑term memory (LSTM)

一.基本信息 论文名称&#xff1a;Intrusion detection systems using longshort‑term memory (LSTM) 中文翻译&#xff1a;基于长短期记忆(LSTM)的入侵检测系统 DOI&#xff1a;10.1186/s40537-021-00448-4 作者&#xff1a;FatimaEzzahra Laghrissi1* , Samira Douzi2*, Kha…

大数据挖掘期末复习

大数据挖掘 数据挖掘 数据挖掘定义 技术层面&#xff1a; 数据挖掘就是从大量的、不完全的、有噪声的、模糊的、随机的实际应用数据中&#xff0c;提取隐含在其中、人们事先不知道的、但又潜在有用的信息的过程。 数据准备环节 数据选择 质量分析 数据预处理 数据仓库 …

搭建es环境

centos7搭建elasticsearch环境 首先考虑使用 Docker 来安装 Elasticsearch、Kibana 和 Logstash。在安装过程中&#xff0c;可能会遇到一些问题&#xff0c;但通过适当的方法可以解决。 docker pull docker.elastic.co/elasticsearch/elasticsearch:8.14.3 首先创建一个网络&a…

k8s 学习笔记之 k8s 存储管理

文章目录 概述卷卷的常用类型emptyDir边车容器 HostPathnfsPV/PVC静态供给 PV 和 PVC创建静态 PV创建 pvc创建 pod 应用 pvc 动态供给 PV 和 PVC创建 StorageClass创建 pvc创建 pod 使用 pvc PV 的生命周期 内置存储对象ConfigMapSecret 配置文件自动重新加载方案**1. 应用内动…

等精度频率计的设计

目录 主控电路设计 频率测量与计算电路设计 顶层电路设计 功能扩展及应用 频率测量的三种方法 等精度频率计通过控制闸门信号与被测信号同步&#xff0c;消除了直接测频法中的计数误差&#xff0c;因而在被测信号频率范围内测量精度基本上是恒定的。 本节以设计能够测量信号…

Matlab使用深度网络设计器为迁移学习准备网络

迁移学习通过对预训练网络进行微调&#xff0c;使深度学习模型能在少量数据下快速适应新任务&#xff0c;类似于“举一反三”&#xff0c;而不需要从头训练。本文使用matlab自带的深度网络设计器&#xff0c;可以便捷地修改预训练网络进行迁移学习&#xff0c;通过对预训练网络…

基于yolov8、yolov5的电塔缺陷检测识别系统(含UI界面、训练好的模型、Python代码、数据集)

摘要&#xff1a;电塔缺陷检测在电力设备巡检、运行维护和故障预防中起着至关重要的作用&#xff0c;不仅能帮助相关部门实时监测电塔运行状态&#xff0c;还为智能化检测系统提供了可靠的数据支撑。本文介绍了一款基于YOLOv8、YOLOv5等深度学习框架的电塔缺陷检测模型&#xf…

蒙特卡洛模拟 详解

蒙特卡洛模拟详解 蒙特卡洛模拟&#xff08;Monte Carlo Simulation&#xff09; 是一种利用随机抽样和统计计算来模拟复杂系统或过程的方法。它尤其适用于分析不确定性、复杂数学问题以及概率分布模型。 蒙特卡洛模拟的基本原理 蒙特卡洛模拟的核心思想是通过 重复随机采样 …

代理商培训新策略:内部知识库的高效运用

在竞争激烈的市场环境中&#xff0c;代理商作为企业与终端消费者之间的关键纽带&#xff0c;其专业能力和服务质量直接影响着企业的市场表现和品牌形象。因此&#xff0c;如何对代理商进行高效、系统的培训&#xff0c;以提升其业务能力和服务水平&#xff0c;成为企业亟需解决…

C++手写PCD文件

前言 一般pcd读写只需要调pcl库接口&#xff0c;直接用pcl的结构写就好了 这里是不依赖pcl库的写入方法 主要是开头写一个header 注意字段大小&#xff0c;类型不要写错     结构定义 写入点需要与header中定义一致 这里用的RoboSense的结构写demo 加了个1字节对齐 stru…

Spring 框架中哪些接口可以创建对象

Spring 框架中哪些接口可以创建对象 在 Spring 框架中&#xff0c;向 IOC 容器中添加 Bean 主要有以下几种接口和方式。Spring 提供了不同的手段来实现对象的创建和管理&#xff0c;涵盖了不同的需求和场景。以下是几种常用的接口和方式&#xff1a; 1. BeanFactory 接口 Be…

uniapp 相关的swiper的一些注意事项

先推荐一个一个对标pc端swiper的uniapp版本 zebra-swiper 缺点是自定义分页器不是很好处理 不知道怎么弄 优点:可以进行高度自适应 &#xff08;这个uniapp原生swiper没有 只能动态修改 采用js 或者只有几种固定高度时采用变量修改&#xff09; <swiperref"lifeMiddle…

ARM中ZI-data段和RW-data段

ARM中ZI-data段和RW-data段 1、只定义全局变量&#xff0c;不使用&#xff0c;不占用内存空间2、 定义并初始化全局变量为0 占用ZI-Data区域3、定义并初始化全局变量非0 占用RW-Data区域4、增加的是一个int8的数据为什么&#xff0c;size增加不是15、定义的全局变量为0&#xf…

jmeter--CSV数据文件设置--请求体设置变量

目录 一、示例 1、准备组织列表的TXT文件&#xff0c;如下&#xff1a; 2、添加 CSV数据文件设置 &#xff0c;如下&#xff1a; 3、接口请求体设置变量&#xff0c;如下&#xff1a; 二、CSV数据文件设置 1、CSV Data Set Config 配置选项说明 2、示例 CSV 文件内容 3、…

golang实现TCP服务器与客户端的断线自动重连功能

1.服务端 2.客户端 生成服务端口程序: 生成客户端程序: 测试断线重连: 初始连接成功

ssm148基于Spring MVC框架的在线电影评价系统设计与实现+jsp(论文+源码)_kaic

毕 业 设 计&#xff08;论 文&#xff09; 题目&#xff1a;在线电影评价系统设计与实现 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本在线电影评价系…

DAY1 网络编程(TCP客户端服务器)

作业&#xff1a; TCP客户端服务器。 server服务器代码&#xff1a; #include <myhead.h> #define IP "192.168.110.52" #define PORT 8886 #define BACKLOG 20 int main(int argc, const char *argv[]) {int oldfdsocket(AF_INET,SOCK_STREAM,0);//IPV4通信…

基于arduino 用ESP8266获取实时MAX30102 血氧数据动态曲线显示在网页上

基于arduino 用ESP8266获取实时MAX30102 血氧数据动态曲线显示在网页上 原理&#xff1a; ESP8266获取MAX30102 血氧数据&#xff08;R,IR,G的值&#xff09;发送到路由器局域网内&#xff0c;局域网内的手机电脑&#xff0c;访问ESP的ip地址&#xff0c;获取实时的血氧数据动…

OpenCV DNN

OpenCV DNN 和 PyTorch 都是常用的深度学习框架&#xff0c;但它们的定位、使用场景和功能有所不同。让我们来对比一下这两个工具&#xff1a; 1. 框架和功能 OpenCV DNN&#xff1a;OpenCV DNN 模块主要用于加载和运行已经训练好的深度学习模型&#xff0c;支持多种深度学习…

vue3:scss引用

原文查看&#xff1a;https://mp.weixin.qq.com/s?__bizMzg3NTAzMzAxNA&mid2247484356&idx2&sn44b127cd394e217b9e3c4eccafdc0aa9&chksmcec6fb1df9b1720b7bd0ca0b321bf8a995fc8cba233deb703512560cbe451cfb1f05cdf129f6&token1776233257&langzh_CN#rd…