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

Gin CORS 跨域请求资源共享与中间件

Gin CORS 跨域请求资源共享与中间件

文章目录

  • Gin CORS 跨域请求资源共享与中间件
    • 一、同源策略
      • 1.1 什么是浏览器的同源策略?
      • 1.2 同源策略判依据
      • 1.3 跨域问题三种解决方案
    • 二、CORS:跨域资源共享简介(后端技术)
    • 三 CORS基本流程
      • 1.CORS请求分类
      • 2.基本流程
    • 四、CORS两种请求详解
      • 1.两种请求详解
      • 2.解决跨域问题:浏览器对于这两种请求的处理
    • 五、Gin 中间件
      • 5.1 中间件介绍
      • 5.2 初识中间件
      • 5.3 c.Next()
      • 5.4 多个中间件执行顺序
      • 5.5 c.Abort()
      • 5.6 全局中间件与局部中间件
      • 5.7 中间件和视图函数之间共享数据
      • 5.8 在路由分组中配置中间件
      • 5.9 中间件解决跨域
      • 5.10 中间件注意事项
        • 5.10.1 Gin 默认中间件
        • 5.10.2 gin中间件中使用 **goroutine**
    • 六、Gin 框架跨域问题解决
      • 5.1 安装
      • 5.2 导入
      • 5.3 直接设置跨域参数(一般用这个)
      • 5.4 使用DefaultConfig作为起点
      • 5.5 Default()允许所有来源

一、同源策略

1.1 什么是浏览器的同源策略?

  • 同源策略(Same origin policy)是一种约定,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,则浏览器的正常功能可能都会受到影响。可以说Web是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现
  • 浏览器最基本的安全策略
  • 浏览器只能接收相同域(IP地址+端口)返回的数据

1.2 同源策略判依据

请求的url地址,必须与浏览器上的url地址处于同域上,也就是域名,端口,协议相同只要协议、域名和端口任意一个不同,都是跨域请求

  1. 比如: 我在本地上的域名是127.0.0.1:8000,请求另外一个域名:127.0.0.1:8001一段数据
  2. 浏览器上就会报错,这个就是同源策略的保护,如果浏览器对javascript没有同源策略的保护,那么一些重要的机密网站将会很危险
  3. 已拦截跨源请求:同源策略禁止读取位于http://127.0.0.1:8001/SendAjax/的远程资源。(原因:CORS 头缺少 'Access-Control-Allow-Origin')
  4. 但是注意,项目2中的访问已经发生了,说明是浏览器对非同源请求返回的结果做了拦截

所以就导致了向不同域发请求,就会出现跨域问题(被浏览器阻止了),正常来说,如果我们不做额外处理,是没办法这么发请求的。

1.3 跨域问题三种解决方案

  • CORS(跨域资源共享:后端技术),主流采用的方案,使用第三方插件
  • 前端代理(只能在测试阶段使用):node起了一个服务,正向代理
  • jsonp:只能解决get请求跨域,本质原理使用了某些标签不限制跨域(img,script)

二、CORS:跨域资源共享简介(后端技术)

CORS需要浏览器和服务器同时支持。目前,所有浏览器都支持该功能,IE浏览器不能低于IE10。

整个CORS通信过程,都是浏览器自动完成,不需要用户参与。对于开发者来说,CORS通信与同源的AJAX通信没有差别,代码完全一样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感觉。

因此,实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。

三 CORS基本流程

1.CORS请求分类

  • 简单请求 (simple request):简单请求只发一次

  • 非简单请求 (not-so-simple request):发送两次,第一次是options请求,第二次是真正的请求

2.基本流程

  • 浏览器发出CORS简单请求只需要在头信息之中增加一个Origin字段。
  • 浏览器发出CORS非简单请求会在正式通信之前,增加一次HTTP查询请求,称为"预检"请求(preflight)。浏览器先询问服务器,当前网页所在的域名是否在服务器的许可名单之中,以及可以使用哪些HTTP动词和头信息字段。只有得到肯定答复,浏览器才会发出正式的XMLHttpRequest请求,否则就报错。

四、CORS两种请求详解

1.两种请求详解

只要同时满足以下两大条件,就属于简单请求

  1. 请求方法是以下三种方法之一:
    • HEAD
    • GET
    • POST
  2. HTTP的头信息不超出以下几种字段:
    • Accept
    • Accept-Language
    • Content-Language
    • Last-Event-ID
    • Content-Type:只限于三个值application/x-www-form-urlencodedmultipart/form-data、text/plain

凡是不同时满足上面两个条件,就属于非简单请求

浏览器对这两种请求的处理,是不一样的。

简单请求和非简单请求的区别

  • 简单请求: 一次请求
  • 非简单请求:两次请求,在发送数据之前会先发一次请求用于做“预检”,只有“预检”通过后才再发送一次请求用于数据传输。

关于“预检”

  • 请求方式:OPTIONS
  • “预检”其实做检查,检查如果通过则允许传输数据,检查不通过则不再发送真正想要发送的消息

如何“预检” ?

  • 如果复杂请求是PUT等请求,则服务端需要设置允许某请求,否则“预检”不通过Access-Control-Allow-Methods

  • 如果复杂请求设置了请求头,则服务端需要设置允许某请求头,否则“预检”不通过Access-Control-Allow-Headers

2.解决跨域问题:浏览器对于这两种请求的处理

支持跨域,简单请求

  • 服务器设置响应头:Access-Control-Allow-Origin = ‘域名’ 或 ‘*’

支持跨域,复杂请求

非简单请求需要判断是否是options请求

由于复杂请求时,首先会发送“预检”请求,如果“预检”成功,则发送真实数据。

  • “预检”请求时,允许请求方式则需服务器设置响应头:Access-Control-Allow-Methods
  • “预检”请求时,允许请求头则需服务器设置响应头: Access-Control-Allow-Headers

五、Gin 中间件

在Gin框架中,中间件(Middleware)是一种允许在请求到达处理程序之前或之后执行一些逻辑的机制。中间件允许你在处理请求的过程中插入一些代码,例如验证请求、记录日志、处理跨域等。

5.1 中间件介绍

中间件是Gin框架的一个关键概念。它是一个函数,接受gin.Context作为参数,可以在请求到达处理程序之前或之后执行一些逻辑。中间件允许你在请求处理过程中执行预处理或后处理的操作。

5.2 初识中间件

在Gin框架中,使用Use方法可以注册一个全局中间件,它将应用于所有路由。例如:

package mainimport ("fmt""github.com/gin-gonic/gin""net/http"
)func LoggerMiddleware(c *gin.Context) {// 在请求处理之前执行的逻辑fmt.Println("Start Logging")// 将请求传递给下一个处理程序c.Next()// 在请求处理之后执行的逻辑fmt.Println("End Logging")
}func main() {r := gin.Default()// 注册全局中间件r.Use(LoggerMiddleware)// 定义路由r.GET("/", func(c *gin.Context) {c.String(http.StatusOK, "Hello, Gin!")})r.Run(":8080")
}

在上述例子中,LoggerMiddleware是一个简单的中间件,用于记录请求日志。通过使用Use方法,我们将这个中间件注册为全局中间件,它将应用于所有的路由。

查看Use方法源码如下:

综上,所以中间件必须是一个 gin.HandlerFunc 类型,配置路由的时候可以传递多个 func 回调函数。

5.3 c.Next()

在中间件中,通过调用c.Next()可以将请求传递给下一个处理程序。这是一个重要的步骤,如果你忘记调用c.Next(),那么请求将不会继续传递给后续的中间件或路由处理程序。

package mainimport ("fmt""github.com/gin-gonic/gin""time"
)// 写一个中间件,统计视图函数运行时间
func totalTime(c *gin.Context) {start := time.Now()c.Next() //继续往后执行end := time.Now()fmt.Println("视图函数运行时间为:", end.Sub(start))
}
func main() {r := gin.Default()// 把中间件函数加在视图函数之前r.GET("/", totalTime, func(c *gin.Context) {time.Sleep(time.Second * 2)c.String(200, "hello,Gin!")})r.Run()
}

5.4 多个中间件执行顺序

如果你有多个中间件,它们将按照注册的顺序执行。在上述例子中,如果我们有多个全局中间件,它们将按照注册的顺序依次执行。

func Middleware1(c *gin.Context) {fmt.Println("Middleware 1 - Start")c.Next()fmt.Println("Middleware 1 - End")
}func Middleware2(c *gin.Context) {fmt.Println("Middleware 2 - Start")c.Next()fmt.Println("Middleware 2 - End")
}func main() {r := gin.Default()// 注册全局中间件,按照注册顺序执行r.Use(Middleware1)r.Use(Middleware2)r.GET("/hello", func(c *gin.Context) {c.String(http.StatusOK, "Hello, Gin!")})r.Run(":8080")
}

输出结果如下:

Middleware 1 - Start
Middleware 2 - Start
Middleware 2 - End
Middleware 1 - End

5.5 c.Abort()

在中间件中,通过调用c.Abort()可以终止请求链,不再执行后续的中间件或路由处理程序。这通常是在中间件中检测到错误或条件不满足时使用的。

package mainimport ("github.com/gin-gonic/gin""net/http"
)func isValidAuth(authorizationHeader string) bool {// 在这里实现你的身份验证逻辑// 例如,你可能验证一个令牌或检查凭证// 如果身份验证成功,返回 true;否则返回 falsereturn true
}func AuthMiddleware(c *gin.Context) {// 检查是否有有效的 Authorization 头if authorizationHeader := c.GetHeader("Authorization"); authorizationHeader == "" {// 如果 Authorization 头缺失,返回未授权状态c.AbortWithStatus(http.StatusUnauthorized)return}// 检查使用你的自定义逻辑提供的身份验证是否有效if !isValidAuth(c.GetHeader("Authorization")) {// 如果身份验证失败,返回未授权状态c.AbortWithStatus(http.StatusUnauthorized)return}// 如果身份验证成功,继续执行下一个中间件或路由处理程序c.Next()
}func main() {r := gin.Default()// 应用 AuthMiddleware 以保护 "/protected" 路由r.GET("/protected", AuthMiddleware, func(c *gin.Context) {// 只有在 AuthMiddleware 允许请求继续时,才会执行此处理程序c.String(http.StatusOK, "你有权访问受保护的路由!")})r.Run(":8080")
}

5.6 全局中间件与局部中间件

全局中间件是通过Use方法注册的,它将应用于所有路由。局部中间件是在定义单个路由时附加的。

package mainimport ("fmt""github.com/gin-gonic/gin""net/http"
)// GlobalMiddleware1 全局中间件1
func GlobalMiddleware1(c *gin.Context) {c.Header("X-Global-Middleware-1", "Applied")fmt.Printf("GlobalMiddleware1\n")c.Next()
}// GlobalMiddleware2 全局中间件2
func GlobalMiddleware2(c *gin.Context) {c.Header("X-Global-Middleware-2", "Applied")fmt.Printf("GlobalMiddleware2\n")c.Next()
}// LocalMiddleware3 局部中间件3
func LocalMiddleware3(c *gin.Context) {c.Header("X-Local-Middleware-3", "Applied")fmt.Printf("LocalMiddleware3\n")c.Next()
}func main() {// 创建一个新的 Gin 引擎r := gin.New()// 使用全局中间件1r.Use(GlobalMiddleware1)// 使用全局中间件2r.Use(GlobalMiddleware2)r.GET("/", func(c *gin.Context) {c.String(http.StatusOK, "Hello,Gin!")})// 定义一个路由组,并在路由组中使用局部中间件3apiGroup := r.Group("/api")apiGroup.Use(LocalMiddleware3)// 路由1,将同时应用全局中间件1、全局中间件2以及局部中间件3apiGroup.GET("/endpoint1", func(c *gin.Context) {c.String(http.StatusOK, "Endpoint 1")})// 路由2,将同时应用全局中间件1、全局中间件2r.GET("/endpoint2", func(c *gin.Context) {c.String(http.StatusOK, "Endpoint 2")})// 启动服务器r.Run(":8080")
}

5.7 中间件和视图函数之间共享数据

在中间件和视图函数之间共享数据可以使用c.Setc.Get方法。

package mainimport ("fmt""github.com/gin-gonic/gin""net/http"
)func CustomMiddleware(c *gin.Context) {// 在中间件中设置数据c.Set("userID", 123)// 继续执行下一个中间件或路由处理程序c.Next()
}func ProtectedRouteHandler(c *gin.Context) {// 从上一个中间件中获取数据userID, exists := c.Get("userID")if !exists {// 如果数据不存在,返回错误响应c.String(http.StatusInternalServerError, "无法获取用户信息")return}// 数据存在,继续处理c.String(http.StatusOK, fmt.Sprintf("用户ID:%v,你有权访问受保护的路由!", userID))
}func main() {r := gin.Default()// 应用 CustomMiddleware 中间件以设置数据r.Use(CustomMiddleware)// 设置保护路由r.GET("/protected", ProtectedRouteHandler)r.Run(":8080")
}

5.8 在路由分组中配置中间件

在Gin框架中,你可以使用路由分组将中间件应用于一组相关的路由。这样,你可以更清晰地组织你的中间件和路由。

func LoggerMiddleware(c *gin.Context) {fmt.Println("Request Log")c.Next()
}func AuthMiddleware(c *gin.Context) {// 检查是否有有效的身份验证信息if !isValidAuth(c.GetHeader("Authorization")) {c.AbortWithStatus(http.StatusUnauthorized)return}c.Next()
}func main() {r := gin.Default()// 创建一个路由分组,并将中间件应用于该分组中的所有路由apiGroup := r.Group("/api", LoggerMiddleware, AuthMiddleware)apiGroup.GET("/users", func(c *gin.Context) {c.String(http.StatusOK, "List of Users")})apiGroup.GET("/products", func(c *gin.Context) {c.String(http.StatusOK, "List of Products")})r.Run(":8080")
}

5.9 中间件解决跨域

下面是一个简单的例子:

package mainimport ("github.com/gin-gonic/gin""net/http"
)func main() {r := gin.Default()// 使用中间件处理跨域问题r.Use(CORSMiddleware())// 其他路由注册r.GET("/hello", func(c *gin.Context) {c.JSON(http.StatusOK, gin.H{"message": "Hello, CORS is enabled!"})})// 启动 Gin 服务器r.Run(":8080")
}// CORSMiddleware 中间件处理跨域问题
func CORSMiddleware() gin.HandlerFunc {return func(c *gin.Context) {c.Writer.Header().Set("Access-Control-Allow-Origin", "*")c.Writer.Header().Set("Access-Control-Allow-Credentials", "true")c.Writer.Header().Set("Access-Control-Allow-Methods", "POST, GET, OPTIONS, PUT, DELETE")c.Writer.Header().Set("Access-Control-Allow-Headers", "Origin, Content-Type, Content-Length, Accept-Encoding, X-CSRF-Token, Authorization")if c.Request.Method == "OPTIONS" {c.AbortWithStatus(204)return}c.Next()}
}

在上述例子中,CORSMiddleware函数返回一个用于处理跨域的中间件。通过将该中间件注册到Gin框架中,可以轻松地解决跨域问题。

5.10 中间件注意事项

5.10.1 Gin 默认中间件

gin.Default()默认使用了 LoggerRecovery 中间件,其中:

  • Logger 中间件将日志写入 gin.DefaultWriter,即使配置了 GIN_MODE=release

  • Recovery 中间件会 recover 任何 panic。如果有 panic 的话,会写入 500 响应码。

    如果不想使用上面两个默认的中间件,可以使用 gin.New()新建一个没有任何默认中间件的 路由。

5.10.2 gin中间件中使用 goroutine

当在中间件或 handler 中启动新的 goroutine 时,不能使用原始的上下文(c *gin.Context), 必须使用其只读副本(c.Copy())

r.GET("/", func(c *gin.Context) {cCp := c.Copy()go func() {// simulate a long task with time.Sleep(). 5 seconds time.Sleep(5 * time.Second)// 这里使用你创建的副本fmt.Println("Done! in path " + cCp.Request.URL.Path)}()c.String(200, "首页")})

六、Gin 框架跨域问题解决

目前大多数的 Web 框架,都提供了 CORS 的解决方案。Gin 也不例外,Gin 里面也提供了一个 middleware 实现来解决跨域问题,打开如下Gin 中间件插件库合集:

https://github.com/gin-gonic/contrib

找到cors 文件夹:

进入后会提示:此软件包已于2016-12-07放弃。请改用gin-contrib/cors。点击进入最新的即可。

  • GitHub 地址:https://github.com/gin-contrib/cors

5.1 安装

go get github.com/gin-contrib/cors

5.2 导入

import "github.com/gin-contrib/cors"

5.3 直接设置跨域参数(一般用这个)

package mainimport ("github.com/gin-contrib/cors""github.com/gin-gonic/gin""strings""time"
)func main() {// 创建一个默认的 Gin 实例server := gin.Default()// 使用 CORS 中间件处理跨域问题,配置 CORS 参数server.Use(cors.New(cors.Config{// 允许的源地址(CORS中的Access-Control-Allow-Origin)// AllowOrigins: []string{"https://foo.com"},// 允许的 HTTP 方法(CORS中的Access-Control-Allow-Methods)AllowMethods: []string{"PUT", "PATCH"},// 允许的 HTTP 头部(CORS中的Access-Control-Allow-Headers)AllowHeaders: []string{"Origin"},// 暴露的 HTTP 头部(CORS中的Access-Control-Expose-Headers)ExposeHeaders: []string{"Content-Length"},// 是否允许携带身份凭证(CORS中的Access-Control-Allow-Credentials)AllowCredentials: true,// 允许源的自定义判断函数,返回true表示允许,false表示不允许AllowOriginFunc: func(origin string) bool {if strings.HasPrefix(origin, "http://localhost") {// 允许你的开发环境return true}// 允许包含 "yourcompany.com" 的源return strings.Contains(origin, "yourcompany.com")},// 用于缓存预检请求结果的最大时间(CORS中的Access-Control-Max-Age)MaxAge: 12 * time.Hour,}))// 启动 Gin 服务器,监听在 0.0.0.0:8080 上server.Run(":8080")
}

5.4 使用DefaultConfig作为起点

func main() {router := gin.Default()// - No origin allowed by default// - GET,POST, PUT, HEAD methods// - Credentials share disabled// - Preflight requests cached for 12 hoursconfig := cors.DefaultConfig()config.AllowOrigins = []string{"http://google.com"}// config.AllowOrigins = []string{"http://google.com", "http://facebook.com"}// config.AllowAllOrigins = truerouter.Use(cors.New(config))router.Run()
}

**注意:**虽然 Default() 允许所有来源,但 DefaultConfig() 不允许,您仍然必须使用 AllowAllOriins

5.5 Default()允许所有来源

func main() {router := gin.Default()// same as// config := cors.DefaultConfig()// config.AllowAllOrigins = true// router.Use(cors.New(config))router.Use(cors.Default())router.Run()
}

使用所有来源会禁用 Gin 为客户端设置 cookie 的能力。处理凭据时,不要允许所有来源。

相关文章:

Gin CORS 跨域请求资源共享与中间件

Gin CORS 跨域请求资源共享与中间件 文章目录 Gin CORS 跨域请求资源共享与中间件一、同源策略1.1 什么是浏览器的同源策略?1.2 同源策略判依据1.3 跨域问题三种解决方案 二、CORS:跨域资源共享简介(后端技术)三 CORS基本流程1.CORS请求分类2.基本流程 四、CORS两种…...

TS:.d.ts 文件 和 declare 的作用

1 declare 做外部声明1.1 声明外部类型1.2 声明外部模块1.2.1 解决引入资源模块报错1.2.2 跳过对第三方库的类型检查 1.3 声明外部变量1.4 声明外部命名空间(作用域) 2 .d.ts 文件做外部声明3 declare global {} 在模块中做外部声明 先说一下我对 .d.ts文…...

JavaScript-jQuery2-笔记

1.获取元素文本、属性、内部结构、表单中的值 获取标签中所夹的文本内容:text() 获取标签的属性值:prop(属性名) 获取表单元素的内容:如 文本框中的内容 val() 获取元素的内部html结构:html() 2.筛选选择器 筛选选择器&#xff1…...

设计模式之多线程版本的if------Balking模式

系列文章目录 设计模式之避免共享的设计模式Immutability(不变性)模式 设计模式之并发特定场景下的设计模式 Two-phase Termination(两阶段终止)模式 设计模式之避免共享的设计模式Copy-on-Write模式 设计模式之避免共享的设计模…...

mybatis核心配置文件介绍

mybatis核心配置文件 1. properties配置介绍 properties标签:加载外部的资源配置文件 ​ 属性:resource 指定要引入的配置文件路径 ​ 在核心配置文件中,通过:${key}方式引入外部配置文件的数据 jdbc.peroperties 的文件内容…...

Linux完全卸载Anaconda3和MiniConda3

如何安装Anaconda3和MiniConda3请看这篇文章: 安装Anaconda3和MiniConda3_minianaconda3-CSDN博客文章浏览阅读474次。MiniConda3官方版是一款优秀的Python环境管理软件。MiniConda3最新版只包含conda及其依赖项如果您更愿意拥有conda以及超过720个开源软件包&…...

Apache Answer,最好的开源问答系统

Apache Answer是一款适合任何团队的问答平台软件。无论是社区论坛、帮助中心还是知识管理平台,你可以永远信赖 Answer。 目前该项目在github超过10K星,系统采用go语言开发,安装配置简单,界面清洁易用,且开源免费。项目…...

【C】内存分配

首先,回顾一下内存分配。所有程序都必须预留足够的内存来存储程序使用的数据。这些内存中有些是自动分配的: float x; int place[100]; 这些声明预留了足够的空间,还为内存提供了一个标识符,可以使用x或place识别数据。 1、mal…...

MySQL 从零开始:03 基本入门语句

文章目录 1、连接数据库1.1 命令提示符登陆1.2 MySQL 8.0 Command Line Client 登陆1.3 MySQL Workbench 登陆 2、基本语句2.1 查看所有库2.2 创建库2.3 删除库2.4 选择数据库2.5 查看表2.6 创建表2.7 删除表2.8 改表名2.9 清空表 在上一小节中介绍了 MySQL 数据库的安装&#…...

井盖异动传感器,守护脚下安全

随着城市化进程的加速,城市基础设施的安全问题日益受到关注。其中,井盖作为城市地下管道的重要入口,其安全问题不容忽视。然而,传统的井盖监控方式往往存在盲区,无法及时发现井盖的异常移动。为此,我们推出…...

复合机器人作为一种新型的智能制造装备高效、精准和灵活的生产方式

随着汽车制造业的快速发展,对于高效、精准和灵活的生产方式需求日益增强。复合机器人作为一种新型的智能制造装备,以其独特的优势在汽车制造中发挥着越来越重要的作用。因此,富唯智能顺应时代的发展趋势,研发出了ICR系列的复合机器…...

重置 Docker 中 Gitlab 的账号密码

1、首先进入Docker容器 docker exec -it gitlab bash 2、连接到 gitlab 的数据库 需要谨慎操作 gitlab-rails console -e production 等待加载完后会进入控制台 ------------------------------------------------------------------------------------------------------…...

任务类型划分

以下内容来自于ChatGPT内存密集型应用和IO密集型应用是两种不同类型的计算应用,它们在资源需求和性能特点上有所不同。 内存密集型应用(Memory-Intensive Applications): 特点: 这类应用主要依赖大量的内存资源来执行任…...

docker搭建部署mysql并挂载指定目录

Docker是一种轻量级、可移植的容器化平台,可以简化应用程序的部署和管理。在本文中,我们将探讨如何使用Docker来搭建和部署MySQL数据库,并将数据和配置文件挂载到外部目录,以实现数据持久化和方便的配置管理。 1: 安装Docker 首…...

即将推出的 OpenWrt One/AP-24.XY:OpenWrt 和 Banana Pi 合作路由器板

OpenWrt开发人员正在与Banana Pi合作开发OpenWrt One/AP-24.XY路由器板。OpenWrt 是一个轻量级嵌入式 Linux 操作系统,支持近 1,800 个路由器和其他设备。然而,这将是第一块由 OpenWrt 直接开发的路由器板。 该主板将基于 MediaTek MT7981B (Filogic 82…...

【uniapp-小程序-分享图5/4】

utils.js //裁剪分享的图片为5:4 const makeCanvas (imgUrl) > {console.log("imgUrl",imgUrl);return new Promise((resolve, reject) > {// 获取图片信息,小程序下获取网络图片信息需先配置download域名白名单才能生效uni.getImageInfo({src: imgUrl,succe…...

【响应式编程】前置知识和相关技术的总结

前置知识 这些概念都与响应式编程密切相关。🦌 1. 并发和多线程编程:响应式编程需要处理并发性,它允许多个操作独立地并行执行。这使得应用程序可以在不同的线程、进程或设备上处理多个事件。 2. 事件驱动编程:响应式编程是一种…...

K8S--安装MySQL8(单机)

原文网址:K8S--安装MySQL8(单机)-CSDN博客 简介 本文介绍K8S部署MySQL8(单机)的方法。 ----------------------------------------------------------------------------------------------- 分享Java真实高频面试题…...

CMake+QT+大漠插件的桌面应用开发

文章目录 CMakeQT大漠插件的桌面应用开发说明环境项目结构配置编译环境代码 CMakeQT大漠插件的桌面应用开发 说明 在CMake大漠插件的应用开发——处理dm.dll,免注册调用大漠插件中已经说明了如何免注册调用大漠插件,以及做了几个简单的功能调用&#x…...

OpenCV-24双边滤波

一、概念 双边滤波对于图像的边缘信息能够更好的保存。其原理为一个与空间距离相关的高斯函数与一个灰度距离相关的高斯函数相乘。 空间距离:指的是当前点与中心点的欧式距离。空间域的高斯函数及其数学形式为: 其中(xi,yi&…...

龙虎榜——20250610

上证指数放量收阴线,个股多数下跌,盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型,指数短线有调整的需求,大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的:御银股份、雄帝科技 驱动…...

idea大量爆红问题解决

问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...

OpenLayers 可视化之热力图

注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

centos 7 部署awstats 网站访问检测

一、基础环境准备(两种安装方式都要做) bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats&#xff0…...

【第二十一章 SDIO接口(SDIO)】

第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...

汽车生产虚拟实训中的技能提升与生产优化​

在制造业蓬勃发展的大背景下,虚拟教学实训宛如一颗璀璨的新星,正发挥着不可或缺且日益凸显的关键作用,源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例,汽车生产线上各类…...

深入理解JavaScript设计模式之单例模式

目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式(Singleton Pattern&#…...

【机器视觉】单目测距——运动结构恢复

ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛&#xf…...

cf2117E

原题链接&#xff1a;https://codeforces.com/contest/2117/problem/E 题目背景&#xff1a; 给定两个数组a,b&#xff0c;可以执行多次以下操作&#xff1a;选择 i (1 < i < n - 1)&#xff0c;并设置 或&#xff0c;也可以在执行上述操作前执行一次删除任意 和 。求…...

《通信之道——从微积分到 5G》读书总结

第1章 绪 论 1.1 这是一本什么样的书 通信技术&#xff0c;说到底就是数学。 那些最基础、最本质的部分。 1.2 什么是通信 通信 发送方 接收方 承载信息的信号 解调出其中承载的信息 信息在发送方那里被加工成信号&#xff08;调制&#xff09; 把信息从信号中抽取出来&am…...