go : gin http2 push资源

2024-03-26 13:58
文章标签 go 资源 push gin http2

本文主要是介绍go : gin http2 push资源,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景:
使用gin 框架,http2 push资源到server
知识点:
http2 server push
HTTP/2旨在解决HTTP/1.x的许多缺陷。现代web页面使用许多资源:HTML、样式表、脚本、图像等等。在HTTP/1.x中,必须显式地请求这些资源中的每一个。这可能是一个缓慢的过程。浏览器从获取HTML开始,然后在解析和计算页面时逐步了解更多资源。由于服务器必须等待浏览器发出每个请求,因此网络通常处于空闲状态且未充分利用。

为了提高延迟,HTTP/2引入了服务器推送,它允许服务器在显式请求资源之前将资源推送到浏览器。服务器通常知道页面将需要的许多额外资源,并且可以在响应初始请求时开始推送这些资源。这允许服务器充分利用空闲的网络并提高页面加载时间。
在这里插入图片描述
在协议层,HTTP/2服务器推送由push-PROMISE帧驱动。PUSH_PROMISE描述服务器预测浏览器将在不久的将来发出的请求。一旦浏览器收到推送承诺,它就知道服务器将交付资源。如果浏览器稍后发现它需要此资源,它将等待推送完成,而不是发送新请求。这减少了浏览器在网络上等待的时间。

server push 在net/http 包里面实现
Go1.8引入了对从http.Server推送响应的支持。如果正在运行的服务器是HTTP/2服务器,并且传入连接使用HTTP/2,则此功能可用。在任何HTTP处理程序中,都可以通过检查HTTP.ResponseWriter是否实现了新的HTTP.Pusher接口来断言它是否支持服务器推送。

例如,如果服务器知道需要app.js来呈现页面,则处理程序可以在http.Pusher可用时启动推送:

http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {if pusher, ok := w.(http.Pusher); ok {// Push is supported.if err := pusher.Push("/app.js", nil); err != nil {log.Printf("Failed to push: %v", err)}}// ...})

Push调用为/app.js创建一个合成请求,将该请求合成为Push-PROMISE框架,然后将合成请求转发给服务器的请求处理程序,后者将生成Push响应。Push的第二个参数指定了Push-PROMISE中要包含的附加头。例如,如果对/app.js的响应在接受编码上有所不同,那么PUSH_PROMISE应该包含一个接受编码值:

http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {if pusher, ok := w.(http.Pusher); ok {// Push is supported.options := &http.PushOptions{Header: http.Header{"Accept-Encoding": r.Header["Accept-Encoding"],},}if err := pusher.Push("/app.js", options); err != nil {log.Printf("Failed to push: %v", err)}}// ...})

关于http2 push 参考 http2 push
实验代码:

package mainimport ("html/template""log""github.com/gin-gonic/gin"
)var html = template.Must(template.New("https").Parse(`
<html>
<head><title>Https Test</title><script src="/assets/app.js"></script>
</head>
<body><h1 style="color:red;">Welcome, Ginner!</h1>
</body>
</html>
`))func main() {r := gin.Default()r.Static("/assets", "./assets")r.SetHTMLTemplate(html)r.GET("/", func(c *gin.Context) {if pusher := c.Writer.Pusher(); pusher != nil {// use pusher.Push() to do server pushif err := pusher.Push("/assets/app.js", nil); err != nil {log.Printf("Failed to push: %v", err)}}c.HTML(200, "https", gin.H{"status": "success",})})// Listen and Server in https://127.0.0.1:8080r.RunTLS(":8080", "./testdata/server.pem", "./testdata/server.key")
}

这篇关于go : gin http2 push资源的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/848738

相关文章

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

go基础知识归纳总结

无缓冲的 channel 和有缓冲的 channel 的区别? 在 Go 语言中,channel 是用来在 goroutines 之间传递数据的主要机制。它们有两种类型:无缓冲的 channel 和有缓冲的 channel。 无缓冲的 channel 行为:无缓冲的 channel 是一种同步的通信方式,发送和接收必须同时发生。如果一个 goroutine 试图通过无缓冲 channel

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而

【Go】go连接clickhouse使用TCP协议

离开你是傻是对是错 是看破是软弱 这结果是爱是恨或者是什么 如果是种解脱 怎么会还有眷恋在我心窝 那么爱你为什么                      🎵 黄品源/莫文蔚《那么爱你为什么》 package mainimport ("context""fmt""log""time""github.com/ClickHouse/clickhouse-go/v2")func main(

49个权威的网上学习资源网站

艺术与音乐 Dave Conservatoire — 一个完全免费的音乐学习网站,口号是“让每一个人都可以接受世界级的音乐教育”,有视频,有练习。 Drawspace — 如果你想学习绘画,或者提高自己的绘画技能,就来Drawspace吧。 Justin Guitar — 超过800节免费的吉他课程,有自己的app,还有电子书、DVD等实用内容。 数学,数据科学与工程 Codecad

Go Select的实现

select语法总结 select对应的每个case如果有已经准备好的case 则进行chan读写操作;若没有则执行defualt语句;若都没有则阻塞当前goroutine,直到某个chan准备好可读或可写,完成对应的case后退出。 Select的内存布局 了解chanel的实现后对select的语法有个疑问,select如何实现多路复用的,为什么没有在第一个channel操作时阻塞 从而导

Go Channel的实现

channel作为goroutine间通信和同步的重要途径,是Go runtime层实现CSP并发模型重要的成员。在不理解底层实现时,经常在使用中对channe相关语法的表现感到疑惑,尤其是select case的行为。因此在了解channel的应用前先看一眼channel的实现。 Channel内存布局 channel是go的内置类型,它可以被存储到变量中,可以作为函数的参数或返回值,它在r

Go 数组赋值问题

package mainimport "fmt"type Student struct {Name stringAge int}func main() {data := make(map[string]*Student)list := []Student{{Name:"a",Age:1},{Name:"b",Age:2},{Name:"c",Age:3},}// 错误 都指向了最后一个v// a