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
地址处于同域上,也就是域名,端口,协议相同,只要协议、域名和端口任意一个不同,都是跨域请求。
- 比如: 我在本地上的域名是
127.0.0.1:8000
,请求另外一个域名:127.0.0.1:8001
一段数据- 浏览器上就会报错,这个就是同源策略的保护,如果浏览器对
javascript
没有同源策略的保护,那么一些重要的机密网站将会很危险- 已拦截跨源请求:同源策略禁止读取位于
http://127.0.0.1:8001/SendAjax/
的远程资源。(原因:CORS 头缺少 'Access-Control-Allow-Origin')
- 但是注意,项目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.两种请求详解
只要同时满足以下两大条件,就属于简单请求
- 请求方法是以下三种方法之一:
- HEAD
- GET
- POST
- HTTP的头信息不超出以下几种字段:
- Accept
- Accept-Language
- Content-Language
- Last-Event-ID
- Content-Type:只限于三个值
application/x-www-form-urlencoded
、multipart/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.Set
和c.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()
默认使用了 Logger
和 Recovery
中间件,其中:
-
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.筛选选择器 筛选选择器࿱…...

设计模式之多线程版本的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&…...

AI智能分析网关V4:太阳能+4G智慧水库远程可视化智能监管方案
一、背景需求分析 由于水库位置分散的原因,水库视频监控建设在立杆、布线等方面都存在一定的难度,且需要人力、物力的前期投入和后期维护。目前水库的监管存在一定的问题,管理人员工作强度大但管理质量并不高,人为巡检无法实时发…...

第8章-第8节-Java中的文件类File的简单介绍
1、我们已经写过File的相关代码: BufferedReader bf new BufferedReader(new FileReader(new File("aa.txt"))); 2、其实FIle这个类自身也是非常强大的,封装了很多操作文件/目录的方法,今天我们就需要去详细的学习这个类&#x…...

CTF-PWN-沙箱逃脱-【seccomp和prtcl-2】
文章目录 沙箱逃脱prtcl题HITCON CTF 2017 Quals Impeccable Artifact libc 2.24flag文件对应prctl函数检查源码思路exp 沙箱逃脱prtcl题 HITCON CTF 2017 Quals Impeccable Artifact libc 2.24 flag文件 此时的flag文件在本文件夹建一个即可 此时的我设置的flag为 对应prc…...

【Docker篇】使用Docker操作镜像
文章目录 🛸镜像🌺基本操作⭐docker --help⭐docker pull [ 参数 ]⭐docker images⭐docker save -- 导出⭐docker rmi -- 删除⭐docker load -- 导入 🛸镜像 镜像是指在计算机领域中,通过复制和创建一个与原始对象相似的副本的过…...

css宽度适应内容
废话不多说,看如下demo,我需要将下面这个盒子的宽度变成内容自适应 方法有很多,如下 父元素设置display:flex 实现子元素宽度适应内容 如下给父元素设置flex能实现宽度自适应内容 <!DOCTYPE html><html lang"en"><head><meta charset"U…...

粒子物理和原子核物理的理论在模拟和分析电路中的粒子束和辐射效应中的应用
粒子物理和原子核物理的理论可以应用于模拟和分析电路中的粒子束和辐射效应,特别是在粒子加速器和辐射探测器的设计和优化方面。通过这些理论的应用,可以提高加速器和探测器的性能,推动粒子物理和原子核物理的研究进展。粒子物理和原子核物理…...

Opentsdb官方优化文档 - 翻译
文档地址 : Tuning — OpenTSDB 2.4 documentation Tuning As with any database there are many tuning parameters for OpenTSDB that can be used to improve write and read performance. Some of these options are specific to certain backends, others are global. …...

JavaScript深拷贝与浅拷贝的全面解析
🧑🎓 个人主页:《爱蹦跶的大A阿》 🔥当前正在更新专栏:《VUE》 、《JavaScript保姆级教程》、《krpano》 目录 ✨ 前言 ✨ 正文 浅拷贝 对象的浅拷贝 数组的浅拷贝 浅拷贝的问题 深拷贝 什么是深拷贝…...

ESU毅速丨制造企业需不需要建设增材制造中心?
随着科技的不断发展,增材制造技术已经成为制造行业的新宠。越来越多的企业开始考虑建设增材制造中心,以提高生产效率、降低成本、加速产品创新。但是,对于制造企业来说,是否需要建设增材制造中心呢? 首先,我…...

Linux shell编程学习笔记39:df命令
0 前言1 df命令的功能、格式和选项说明 1.1 df命令的功能1.2 df命令的格式1.3 df命令选项说明 2 df命令使用实例 2.1 df:显示主要文件系统信息2.2 df -a:显示所有文件系统信息2.3 df -t[]TYPE或--type[]TYPE:显示TYPE指定类型的文件系统信…...