G0第22章 :获取参数、文件上传、重定向、Gin路由、Gin中间件、运行多个服务

news2024/10/6 13:16:05

05 获取参数

1、获取querystring参数

querystring 指的是URL中的 ? 后面携带的参数,例如: /user/search?username=小王子&address=天津 。获取请求querystring参数的方法如下:

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
)

func main() {
	r := gin.Default()

	// quertsting

	//GET请求 URL ? 后面是querystring的参数
	//key-value格式,多个key-value 用 & 连接
	// eq : http://127.0.0.1:9090/user/search?query=小王子&age=18
	// http://127.0.0.1:9090/user/search?query=     则默认返回为空

	r.GET("/user/search", func(c *gin.Context){
		//获取浏览器那边发请求携带的 query string 参数
		name := c.Query("query") //通过Query获取请求中携带的querystring参数
		age := c.Query("age")
		//name := c.DefaultQuery("query","somebody")//取不到就用指定的默认值
		/*name, ok := c.GetQuery("query")//渠道返回(值,true),取不到第二个参数就返回("", false)
		//if !ok{
		//	//取不到
		//	name = "somebody"
		}*/

		//输出json结果给调用方
		data := gin.H{
			"name": name,
			"age": age,
		}
		c.JSON(http.StatusOK, data)
	})
	r.Run(":9090")
}

请添加图片描述

2、获取form参数

请求的数据通过form表单来提交,例如相 /user/search 发送一个POST请求,获取请求数据的方式如下:

func main() {
	//Default返回一个默认的路由引擎
	r := gin.Default()
	r.POST("/user/search", func(c *gin.Context) {
		// DefaultPostForm取不到值时会返回指定的默认值
		//username := c.DefaultPostForm("username", "小王子")
		username := c.PostForm("username")
		address := c.PostForm("address")
		//输出json结果给调用方
		c.JSON(http.StatusOK, gin.H{
			"message":  "ok",
			"username": username,
			"address":  address,
		})
	})
	r.Run(":8080")
}

示例:

main.go

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
)

// 获取 form 表单提交与获取参数

func main(){
	r := gin.Default()

	r.LoadHTMLFiles("./login.html","./index.html")

	r.GET("/login",func(c *gin.Context){
		c.HTML(http.StatusOK, "login.html", nil)
	})

	r.Run(":9090")
}

login.html

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>login</title>
</head>
<body>
<form action="/login" method="post" novalidate autocomplete="off">
    <div>
        <label for="username">username:</label>
        <input type="text" name="username" id="username">
    </div>

    <div>
        <label for="password">password:</label>
        <input type="password" name="password" id="password">
    </div>

    <div>
        <input type="submit" value="登录">
    </div>

</form>
</body>
</html>

在浏览器用户的一次请求(127.0.0.1:9090/login),
对应后端代码的一次响应
(r.GET(“/login”,func(c *gin.Context){
c.HTML(http.StatusOK, “login.html”, nil)
}))
main.go

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
)

// 获取 form 表单提交与获取参数

func main(){
	r := gin.Default()

	r.LoadHTMLFiles("./login.html","./index.html")
	//一个请求一个响应
	r.GET("/login",func(c *gin.Context){
		c.HTML(http.StatusOK, "login.html", nil)
	})

	// /login post请求
	//一个请求一个响应
	r.POST("/login", func( c *gin.Context){
		// 第一种获取form表单的方法
		//username := c.PostForm("username")
		//password := c.PostForm("password") 取到就返回值,取不到就返回一个空字符串
		// 第二种
		//username := c.DefaultPostForm("username","somebody")
		//password := c.DefaultPostForm("password","******")
		// 第三种
		username, ok := c.GetPostForm("username")
		if !ok{
			username = "somebody"
		}
		password, ok := c.GetPostForm("password")
		if !ok{
			username = "******"
		}
		c.HTML(http.StatusOK, "index.html", gin.H{
			"Name": username,
			"Password":password,
		})
	})


	r.Run(":9090")
}

index.html 模版渲染

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>Index</title>
</head>
<body>
<h1>hello, {{ .Name }}!</h1>
<p>你的密码是: {{ .Password }}</p>
</body>
</html>

在浏览器用户的一次请求(点击登录请求),
对应后端代码的一次响应 (返回一个index.html页面)

 r.POST("/login", func( c *gin.Context){
 username := c.PostForm("username")
password := c.PostForm("password")
c.HTML(http.StatusOK, "index.html", gin.H{
		"Name": username,
		"Password":password,
	})
})

3、获取json参数

当前端请求的数据通过JSON提交时,例如向/json发送一个POST请求,则获取请求参数的方式如下:

r.POST("/json", func(c *gin.Context) {
	// 注意:下面为了举例子方便,暂时忽略了错误处理
	b, _ := c.GetRawData()  // 从c.Request.Body读取请求数据
	// 定义map或结构体
	var m map[string]interface{}
	// 反序列化
	_ = json.Unmarshal(b, &m)

	c.JSON(http.StatusOK, m)
})

更便利的获取请求参数的方式,参见下面的 参数绑定 小节。

4、获取path参数

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
)

// 获取请求中的path (URL) 参数 返回的都是字符串
// 注意URL的匹配不要冲突
func main(){
	r := gin.Default()

	r.GET("/user/:name/:age", func(c *gin.Context){
		// 获取路径参数
		name := c.Param("name")
		age := c.Param("age") //返回的是string类型
		c.JSON(http.StatusOK, gin.H{
			"name": name,
			"age": age,
		})
	})

	r.GET("/blog/:year/:month", func(c *gin.Context) {
		year := c.Param("year")
		month := c.Param("month")
		c.JSON(http.StatusOK, gin.H{
			"year": year,
			"month":month,
		})
	})

	r.Run(":9090")
}

5、参数绑定

为了能够更方便的获取请求相关参数,提高开发效率,我们可以基于请求Content-Type识别请求数据类型并利用反射机制自动提取请求中 QuerString 、 form表单、 JSON、 XML 等参数到结构体中。下面的示例代码演示了 .ShouldBind() 强大的功能,它能够基于请求自动提取 JSON、form表单和QuerSting 类型的数据,并把值绑定到指定的结构体对象。
index.html

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>index</title>
</head>
<body>
<form action="/form" method="post">
    用户名:
    <input type="text" name="username">
    密码:
    <input type="password" name="password">
    <input type="submit" name="提交">
</form>
</body>
</html>

main.go

package main

import (
	"fmt"
	"github.com/gin-gonic/gin"
	"net/http"
)


type UserInfo struct{
	Username string `form:"username" json:"username"`
	Password string `form:"password" json:"password"`
}

func main() {

	r := gin.Default()

	r.LoadHTMLFiles("./index.html")

	r.GET("/user", func(c *gin.Context){
		//username := c.Query("username")
		//password := c.Query("password")
		//u := UserInfo{
		//	username: username,
		//	password: password,
		//}
		var u UserInfo //声明一个UserInfo类型的变量u
		err := c.ShouldBind(&u)
		if err != nil {
			c.JSON(http.StatusBadRequest, gin.H{
				"error": err.Error(),
			})
			return
		}else{
			fmt.Printf("%#v\n", u)
			c.JSON(http.StatusOK, gin.H{
				"status":"OK",
			})
		}

	})

	r.GET("/index", func (c *gin.Context){
		c.HTML(http.StatusOK, "index.html", nil)
	})

	r.POST("/form", func(c *gin.Context) {
		var u UserInfo
		err := c.ShouldBind(&u)
		if err != nil {
			c.JSON(http.StatusBadRequest, gin.H{
				"error": err.Error(),
			})
		}else{
			fmt.Printf("%#v\n",u)
			c.JSON(http.StatusOK, gin.H{
				"status": "ok",
			})
		}
	})


	r.Run(":9090")

}

请添加图片描述
请添加图片描述
ShouldBind 会按照下面的顺序解析请求中的数据完成绑定:

  • 1、如果是GET请求,只使用Form绑定引擎(query)
  • 2、如果是POST请求,首先检查 content-type 是否为 JSON或 XML,然后再使用Form( form-data)

06 文件上传

单个文件上传

文件上传前端页面代码:

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>index</title>
</head>
<body>

<form action="/upload" method="post" enctype="multipart/form-data">

  <input type="file" name="f1">
  <input type="submit" name="提交">

</form>

</body>
</html>

如果服务器端上传的文件过大内存不足以放下,就先将文件放入一个临时文件夹中,一点一点的去读,可以设置内存的最大限制,来控制文件上传的速度。

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
	"path"
)

func main() {
	r := gin.Default()
	// 处理multipart forms提交文件时默认的内存限制是32 MiB
	// 可以通过下面的方式修改
	// router.MaxMultipartMemory = 8 << 20  // 8 MiB
	r.LoadHTMLFiles("./index.html")
	r.GET("/index", func(c *gin.Context) {
		c.HTML(http.StatusOK, "index.html", nil)
	})
	r.POST("/upload", func(c *gin.Context) {
		// 1、从请求中读取文件
		filereader, err := c.FormFile("f1") // 从请求中获取携带的参数是一样的
		if err != nil {
			c.JSON(http.StatusBadRequest, gin.H{
				"error": err.Error(),
			})
		} else {
			// 2、将读取到的文件保存在服务端的本地
			//dst := fmt.Sprintf("./%s", filereader.Filename)
			dst := path.Join("./", filereader.Filename)
			c.SaveUploadedFile(filereader, dst)
			if err != nil {
				c.JSON(http.StatusBadRequest, gin.H{
					"error": err.Error(),
				})
			}
			c.JSON(http.StatusOK, gin.H{
				"status": "ok",
			})
		}

	})
	r.Run(":9090")
}

多个文件上传

package main

import (
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
	"path"
)

func main() {
	router := gin.Default()

	router.LoadHTMLFiles("./index.html")
	router.GET("/index", func(c *gin.Context){
		c.HTML(http.StatusOK, "index.html", nil)
	})
	// 处理multipart forms 提交文件时默认的内存限制是32MiB
	// 可以通过下面的方式修改
	// router.MaxMultipartMemory 8 << 20 //8 MiB
	router.POST("/uploads", func(c *gin.Context){
		// Multipart form
		form, err := c.MultipartForm()
		if err != nil {
			log.Fatal(err)
		}
		// 通过字段名映射
		files := form.File["file"]
		// for range 遍历文件
		for index, file := range files{
			log.Println(file.Filename)
			dst := path.Join("/Users/tianyi/Desktop/tmp", file.Filename, string(index))
			// 上传文件到指定的目录
			c.SaveUploadedFile(file, dst)
		}
		c.JSON(http.StatusOK, gin.H{
			"message": fmt.Sprintf("%v files uploaded!", len(files)),
		})
})
	router.Run(":9090")

}

07 重定向

HTTP重定向

r.GET("/test", func(c *gin.Context){
	c.Redirect(http.StatusMovePermanently, "http://www.sogo.com/")
})

路由重定向

路由重定向,使用 HandleContext:

r.GET("/test", func(c *gin.Context){
	// 指定重定向的URL
	c.Request.URL.Path = "/test2"
	r.HandleContext(c)
})
r.GET("/test2", func(c *gin.Context){
	c.JSON(http.StatusOk, gin.H{
		"hellp"  : "world!"
	})
})

08 Gin路由

普通路由

r.GET("/index", func(c *gin.Context) {...})
r.GET("/login", func(c *gin.Context) {...})
r.POST("/login", func(c *gin.Context) {...})

此外,还有一个可以匹配所有请求方法的 Any 方法如下:

r.Any("/test", func(c *gin.Context) {...} )

为没有配置处理函数的路由添加处理程序,默认情况下它返回404代码,下面的代码为没有匹配到路由的请求都返回 views/404.html 页面。

r.NoRoute(func(c *gin.Context) {
	c.HTML(http.StatusNotFound, "views/404.html", nil)
})

路由组

我们可以将拥有共同URL前缀的路由划分为一个路由组。习惯性一对 {} 包裹同组的路由,这只是为了看着清晰。用不用{} 包裹功能上没什么区别

package main

import (
	"github.com/gin-gonic/gin"
	"net/http"
)

func main() {
	r := gin.Default()
	// 访问/index的GET请求会走这一条处理逻辑
	// 获取信息的路由
	//r.HEAD()
	r.GET("/index", func(c *gin.Context) {
		c.JSON(http.StatusOK, gin.H{
			"method": "GET",
		})
	})
	// 创建信息的路由
	r.POST("/index", func(c *gin.Context) {
		c.JSON(http.StatusOK, gin.H{
			"method": "POST",
		})
	})
	// 删除信息的路由
	r.DELETE("/index", func(c *gin.Context) {
		c.JSON(http.StatusOK, gin.H{
			"method": "DELETE",
		})
	})
	// 更新部分信息的路由
	r.PUT("/index", func(c *gin.Context) {
		c.JSON(http.StatusOK, gin.H{
			"method": "PUT",
		})
	})
	r.Any("/user", func(c *gin.Context) {
		switch c.Request.Method{
		case "GET":
			c.JSON(http.StatusOK, gin.H{"method": "GET",})
		case http.MethodPost:
			c.JSON(http.StatusOK, gin.H{"method": "POST",})
		}
		c.JSON(http.StatusOK, gin.H{
			"method": "Any",
		})
	})
	//NoRoute
	r.NoRoute(func(c *gin.Context) {
		c.JSON(http.StatusNotFound, gin.H{
			"msg":"liwenzhou.com"})
	})
	//路由组
	//把公用的前缀提取出来,创建一个路由组
	// 视频的首页和详情页
	videoGroup := r.Group("/video")
	{
		videoGroup.GET("index", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/video/index",
			})
		})
		videoGroup.GET("xx", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/video/xx",
			})
		})
		videoGroup.GET("oo", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/video/oo",
			})
		})
	}


	// 商城的首页和详情页
	shopGroup := r.Group("/shop")
	{
		shopGroup.GET("/index", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/shop/index",
			})
		})
		shopGroup.GET("xx", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/shop/xx",
			})
		})
		shopGroup.GET("oo", func(c *gin.Context) {
			c.JSON(http.StatusOK, gin.H{
				"msg": "/shop/oo",
			})
		})
		//嵌套
		q := shopGroup.Group("/oo")
		{
			q.GET("/xx", func(c *gin.Context) {
				c.JSON(http.StatusOK, gin.H{
					"msg": "/shop/oo/xx",
				})
				
			})
		}
	}

	r.Run(":9090")
}

Summry:

Any的用处:可以使用多种方法,注意要优先使用go语言定义好的常量

请添加图片描述

NoRoute的用处:用来引导用户进入到正确的路由

请添加图片描述

路由组的用处:用来区分不同的业务线或API版本

请添加图片描述

路由原理(进阶版会讲)

Gin框架中的路由使用的是httprouter这个库

其基本原理就是构造一个路由地址的前缀树

09 Gin中间件

Gin框架允许开发者在处理请求的过程中,加入用户自己的钩子(Hook)函数。这个钩子函数就叫中间件,中间件适合处理一些公共的业务逻辑,比如登录认证、权限校验、数据分页、记录日志、耗时统计等。

请添加图片描述

定义中间件

Gin中的中间件必须是一个 gin.HandlerFunc 类型。

记录接口耗时的中间件

例如我们像下面的代码一样定义一个统计请求耗时的中间件。

针对单个method使用中间件

demo01

package main

import (
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
	"time"
)

// handlerFunc
func indexHandler(c *gin.Context) {
	fmt.Println("index")
	c.JSON(http.StatusOK, gin.H{
		"msg": "index",
	})
}

// StatCost 是一个统计耗时请求耗时的中间件
func m1(c *gin.Context) {
	fmt.Println("m1..in")
	start := time.Now()
	c.Set("name", "小王子") // 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
	//调用该请求的剩余处理程序
	c.Next()
	//不调用该请求的剩余处理程序
	// c.Abort()
	// 计算耗时
	cost := time.Since(start)
	log.Printf("cost time = %v\n", cost)
	fmt.Println("m1..out")
}

func main() {
	r := gin.Default()
	//GET(relativePath string, handlers ...HandlerFunc) IRoutes {
	r.GET("/index", m1, indexHandler)

	r.Run(":9090")
}

针对全局使用中间件

demo02

package main

import (
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
	"time"
)

// handlerFunc
func indexHandler(c *gin.Context) {
	fmt.Println("index")
	c.JSON(http.StatusOK, gin.H{
		"msg": "index",
	})
}

// handlerFunc
func shopHandler(c *gin.Context) {
	fmt.Println("shop")
	c.JSON(http.StatusOK, gin.H{
		"msg": "shop",
	})
}

// handlerFunc
func userHandler(c *gin.Context) {
	fmt.Println("user")
	c.JSON(http.StatusOK, gin.H{
		"msg": "user",
	})
}

// StatCost 是一个统计耗时请求耗时的中间件
func m1(c *gin.Context) {
	fmt.Println("m1..in")
	start := time.Now()
	c.Set("name", "小王子") // 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
	//调用该请求的剩余处理程序
	c.Next()
	//不调用该请求的剩余处理程序
	// c.Abort()
	// 计算耗时
	cost := time.Since(start)
	log.Printf("cost time = %v\n", cost)
	fmt.Println("m1..out")
}

func main() {
	r := gin.Default()
	// 全局使用中间件
	r.Use(m1)
	//GET(relativePath string, handlers ...HandlerFunc) IRoutes {
	r.GET("/index", indexHandler)
	r.GET("/shop", shopHandler)
	r.GET("/user", userHandler)


	r.Run(":9090")
}

r.User(中间件1, 中间件2)
package main

import (
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
	"time"
)

// handlerFunc
func indexHandler(c *gin.Context) {
	fmt.Println("index")
	c.JSON(http.StatusOK, gin.H{
		"msg": "index",
	})
}

// handlerFunc
func shopHandler(c *gin.Context) {
	fmt.Println("shop")
	c.JSON(http.StatusOK, gin.H{
		"msg": "shop",
	})
}

// handlerFunc
func userHandler(c *gin.Context) {
	fmt.Println("user")
	c.JSON(http.StatusOK, gin.H{
		"msg": "user",
	})
}

// m1 是一个统计耗时请求耗时的中间件
func m1(c *gin.Context) {
	fmt.Println("m1..in")
	start := time.Now()
	c.Set("name", "小王子") // 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
	//调用该请求的剩余处理程序
	c.Next()
	//不调用该请求的剩余处理程序
	// c.Abort()
	// 计算耗时
	cost := time.Since(start)
	log.Printf("cost time = %v\n", cost)
	fmt.Println("m1..out")
}

func m2(c *gin.Context) {
	fmt.Println("m2..in")
	//调用该请求的剩余处理程序
	c.Next()

	fmt.Println("m2..out")
}

func main() {
	r := gin.Default()
	// 全局使用中间件
	r.Use(m1, m2)
	//GET(relativePath string, handlers ...HandlerFunc) IRoutes {
	r.GET("/index", indexHandler)
	r.GET("/shop", shopHandler)
	r.GET("/user", userHandler)


	r.Run(":9090")
}

请添加图片描述

请添加图片描述

c.Abort()

请添加图片描述

请添加图片描述
请添加图片描述

在中间件中使用return结束当前中间件,return之后的代码不会被执行

请添加图片描述

请添加图片描述
请添加图片描述

通常将中间件写为一个闭包
doCheck 作为开关控制是否检查

请添加图片描述
请添加图片描述

记录响应体的中间件

我们有时可能会想记录下某些情况下返回给客户端的响应数据,这个时候就可以编写一个中间件来搞定。


type bodyLogWriter struct {
	gin.ResponseWriter               // 嵌入gin框架ResponseWriter
	body               *bytes.Buffer // 我们记录用的response
}

// Write 写入响应体数据
func (w bodyLogWriter) Write(b []byte) (int, error) {
	w.body.Write(b)                  // 我们记录一份
	return w.ResponseWriter.Write(b) // 真正写入响应
}

// ginBodyLogMiddleware 一个记录返回给客户端响应体的中间件
// https://stackoverflow.com/questions/38501325/how-to-log-response-body-in-gin
func ginBodyLogMiddleware(c *gin.Context) {
	blw := &bodyLogWriter{body: bytes.NewBuffer([]byte{}), ResponseWriter: c.Writer}
	c.Writer = blw // 使用我们自定义的类型替换默认的

	c.Next() // 执行业务逻辑

	fmt.Println("Response body: " + blw.body.String()) // 事后按需记录返回的响应
}

跨域中间件cors

推荐使用社区的https://github.com/gin-contrib/cors 库,一行代码解决前后端分离架构下的跨域问题。

在gin框架中,我们可以为每个路由添加任意数量的中间件。
这个库支持各种常用的配置项,具体使用方法如下。

package main

import (
  "time"

  "github.com/gin-contrib/cors"
  "github.com/gin-gonic/gin"
)

func main() {
  router := gin.Default()
  // CORS for https://foo.com and https://github.com origins, allowing:
  // - PUT and PATCH methods
  // - Origin header
  // - Credentials share
  // - Preflight requests cached for 12 hours
  router.Use(cors.New(cors.Config{
    AllowOrigins:     []string{"https://foo.com"},  // 允许跨域发来请求的网站
    AllowMethods:     []string{"GET", "POST", "PUT", "DELETE",  "OPTIONS"},  // 允许的请求方法
    AllowHeaders:     []string{"Origin", "Authorization", "Content-Type"},
    ExposeHeaders:    []string{"Content-Length"},
    AllowCredentials: true,
    AllowOriginFunc: func(origin string) bool {  // 自定义过滤源站的方法
      return origin == "https://github.com"
    },
    MaxAge: 12 * time.Hour,
  }))
  router.Run()
}
当然你可以简单的像下面的示例代码那样使用默认配置,允许所有的跨域请求。
func main() {
  router := gin.Default()
  // same as
  // config := cors.DefaultConfig()
  // config.AllowAllOrigins = true
  // router.Use(cors.New(config))
  router.Use(cors.Default())
  router.Run()
}
注意: 该中间件需要注册在业务处理函数前面。

这个库支持各种常用的配置项,具体使用方法如下。

package main

import (
  "time"

  "github.com/gin-contrib/cors"
  "github.com/gin-gonic/gin"
)

func main() {
  router := gin.Default()
  // CORS for https://foo.com and https://github.com origins, allowing:
  // - PUT and PATCH methods
  // - Origin header
  // - Credentials share
  // - Preflight requests cached for 12 hours
  router.Use(cors.New(cors.Config{
    AllowOrigins:     []string{"https://foo.com"},  // 允许跨域发来请求的网站
    AllowMethods:     []string{"GET", "POST", "PUT", "DELETE",  "OPTIONS"},  // 允许的请求方法
    AllowHeaders:     []string{"Origin", "Authorization", "Content-Type"},
    ExposeHeaders:    []string{"Content-Length"},
    AllowCredentials: true,
    AllowOriginFunc: func(origin string) bool {  // 自定义过滤源站的方法
      return origin == "https://github.com"
    },
    MaxAge: 12 * time.Hour,
  }))
  router.Run()
}

当然你可以简单的像下面的示例代码那样使用默认配置,允许所有的跨域请求。

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框架中,我们可以为每个路由添加任意数量的中间件。

为全局路由注册

func main() {
	// 新建一个没有任何默认中间件的路由
	r := gin.New()
	// 注册一个全局中间件
	r.Use(StatCost())
	
	r.GET("/test", func(c *gin.Context) {
		name := c.MustGet("name").(string) // 从上下文取值
		log.Println(name)
		c.JSON(http.StatusOK, gin.H{
			"message": "Hello world!",
		})
	})
	r.Run()
}

为某个路由单独注册

// 给/test2路由单独注册中间件(可注册多个)
	r.GET("/test2", StatCost(), func(c *gin.Context) {
		name := c.MustGet("name").(string) // 从上下文取值
		log.Println(name)
		c.JSON(http.StatusOK, gin.H{
			"message": "Hello world!",
		})
	})

为路由组注册中间件

为路由组注册中间件有以下两种写法。
写法1:

shopGroup := r.Group("/shop", StatCost())
{
    shopGroup.GET("/index", func(c *gin.Context) {...})
    ...
}

写法2:

shopGroup := r.Group("/shop")
shopGroup.Use(StatCost())
{
    shopGroup.GET("/index", func(c *gin.Context) {...})
    ...
}

从中间件设置值,并在上下文取值(跨中间件取值)

package main

import (
	"bytes"
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
	"time"
)



// handlerFunc
func indexHandler(c *gin.Context) {
	fmt.Println("index")
	name, ok := c.Get("name")// 从上下文中取值 (跨中间件存取值)
	if !ok{
		name = "匿名用户"
	}
	c.JSON(http.StatusOK, gin.H{
		"msg": name,
	})
}

// handlerFunc
func shopHandler(c *gin.Context) {
	fmt.Println("shop")
	c.JSON(http.StatusOK, gin.H{
		"msg": "shop",
	})
}

// handlerFunc
func userHandler(c *gin.Context) {
	fmt.Println("user")
	c.JSON(http.StatusOK, gin.H{
		"msg": "user",
	})
}

// m1 是一个统计耗时请求耗时的中间件
func m1(c *gin.Context) {
	fmt.Println("m1..in")
	start := time.Now()
	//c.Set("name", "小王子") // 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
	//调用该请求的剩余处理程序
	c.Next()
	//不调用该请求的剩余处理程序
	// c.Abort()
	// 计算耗时
	name, ok := c.Get("name")// 从上下文中取值 (跨中间件存取值)
	if !ok{
		name = "匿名用户"
	}
	c.JSON(http.StatusOK, gin.H{
		"msg": name,
	})
	cost := time.Since(start)
	log.Printf("cost time = %v\n", cost)
	fmt.Println("m1..out")
}

func m2(c *gin.Context) {
	fmt.Println("m2..in")
	c.Set("name","qimi")// 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
	c.Next()
	//阻止调用该请求的剩余处理程序
	//c.Abort()
	fmt.Println("m2..out")
}



func authMiddleware(doCheck bool)gin.HandlerFunc{
	//连接数据库
	// 或是一些其他准备工作
	return func(c *gin.Context){
		fmt.Println("authMiddleware..in")
		if doCheck{
			//存放具体的逻辑

			// 是否登录的判断
			//if 登录
			c.Next()
			//else
			//c.Abort()
		}else{
			c.Next()
		}
		fmt.Println("authMiddleware..out")
	}
}

// StatCost 是一个统计耗时请求耗时的中间件
func StatCost() gin.HandlerFunc {
	return func(c *gin.Context) {
		start := time.Now()
		c.Set("name", "小王子") // 可以通过c.Set在请求上下文中设置值,后续的处理函数能够取到该值
		// 调用该请求的剩余处理程序
		c.Next()
		// 不调用该请求的剩余处理程序
		// c.Abort()
		// 计算耗时
		cost := time.Since(start)
		log.Println(cost)
	}
}

type bodyLogWriter struct{
	gin.ResponseWriter   // 嵌入gin框架ResponseWriter
	body *bytes.Buffer //我们记录用的response
}

// Write 写入响应体数据
func (w bodyLogWriter) Write(b []byte) (int, error){
	w.body.Write(b) //我们记录一份
	return w.ResponseWriter.Write(b)//真正写入响应
}
//ginBodyLogMiddleware 一个记录返回给客户端响应体的中间件
func ginBodyLogMiddleware(user bool) gin.HandlerFunc{
	return func(c *gin.Context){
		blw := &bodyLogWriter{
			body: bytes.NewBuffer([]byte{}),
			ResponseWriter: c.Writer,
			}
		c.Writer = blw // 使用我们自定义的类型替换默认的

		c.Next() //业务逻辑

		fmt.Println("Response body: " + blw.body.String()) //事后按需记录返回的响应
	}
}



func main() {
	r := gin.Default()
	// 全局使用中间件
	r.Use(m1, m2, authMiddleware(true), ginBodyLogMiddleware(true))
	//GET(relativePath string, handlers ...HandlerFunc) IRoutes {
	r.GET("/index", indexHandler)
	r.GET("/shop", shopHandler)
	r.GET("/user", userHandler)
	//路由组注册中间件方法1:
	//shopGroup := r.Group("/xx",StatCost())
	//{
	//	shopGroup.GET("/shop", func(c *gin.Context) {
	//		c.JSON(http.StatusOK, gin.H{
	//			"msg": "shopGroup",
	//		})
	//	})
	//}
	路由组注册中间件方法2:
	//indexGroup := r.Group("/xx2")
	//indexGroup.Use(StatCost())
	//{
	//	indexGroup.GET("/index", func(c *gin.Context) {
	//		c.JSON(http.StatusOK, gin.H{
	//			"msg": "indexGroup",
	//		})
	//	})
	//}
	r.Run(":9090")
}

中间件注意事项

gin默认中间件

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

Logger中间件将日志写入gin.DefaultWriter,即使配置了GIN_MODE=release
Recovery中间件会recover任何panic。如果有panic的话,会写入500响应码。
如果不想使用上面两个默认的中间件,可以使用**gin.New()**新建一个没有任何默认中间件的路由。

gin中间件中使用goroutine

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

10 运行多个服务

我们可以在多个端口启动服务,例如:

package main

import (
	"log"
	"net/http"
	"time"

	"github.com/gin-gonic/gin"
	"golang.org/x/sync/errgroup"
)

var (
	g errgroup.Group
)

func router01() http.Handler {
	e := gin.New()
	e.Use(gin.Recovery())
	e.GET("/", func(c *gin.Context) {
		c.JSON(
			http.StatusOK,
			gin.H{
				"code":  http.StatusOK,
				"error": "Welcome server 01",
			},
		)
	})

	return e
}

func router02() http.Handler {
	e := gin.New()
	e.Use(gin.Recovery())
	e.GET("/", func(c *gin.Context) {
		c.JSON(
			http.StatusOK,
			gin.H{
				"code":  http.StatusOK,
				"error": "Welcome server 02",
			},
		)
	})

	return e
}

func main() {
	server01 := &http.Server{
		Addr:         ":8080",
		Handler:      router01(),
		ReadTimeout:  5 * time.Second,
		WriteTimeout: 10 * time.Second,
	}

	server02 := &http.Server{
		Addr:         ":8081",
		Handler:      router02(),
		ReadTimeout:  5 * time.Second,
		WriteTimeout: 10 * time.Second,
	}
   // 借助errgroup.Group或者自行开启两个goroutine分别启动两个服务
	g.Go(func() error {
		return server01.ListenAndServe()
	})

	g.Go(func() error {
		return server02.ListenAndServe()
	})

	if err := g.Wait(); err != nil {
		log.Fatal(err)
	}
}

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

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

相关文章

Git基础笔记(只含常用命令)

1.Git基本概念 1.git三区 工作区 -->git add 暂存区 --> git commit版本区域 暂存区用来存放临时文件&#xff0c;相当于只编辑没有保存 2.文件状态 命令 git status 查看文件状态 大体来说有被版本管理和没被版本管理两种状态 a.txt 执行了git add 3.git本地仓库命令 g…

面试篇:SpringCloud

一、SpringCloud常见的组件有什么&#xff1f; 1、常见微服务功能架构图 2、阿里巴巴SpringCloud常用组件 注册中心/配置中心&#xff1a;Nacos负载均衡&#xff1a;Ribbon服务调用&#xff1a;Feign服务保护&#xff1a;Sentinel服务网关&#xff1a;Gateway 二、服务注册…

阿里云IoT物联网平台

IoT物联网平台 一、工作原理 1、MQTT&#xff08;Message Queuing Telemetry Transport&#xff09; 是一种轻量级的消息传输协议&#xff0c;专门设计用于物联网&#xff08;IoT&#xff09;应用中的通信。它是一种发布/订阅&#xff08;publish/subscribe&#xff09;模式的…

【OpenCV DNN】Flask 视频监控目标检测教程 04

欢迎关注『OpenCV DNN Youcans』系列&#xff0c;持续更新中 【OpenCV DNN】Flask 视频监控目标检测教程 04 3.4 用Flask构建流媒体服务器3.4.1 流媒体服务器基本知识3.4.2 用Flask搭建流媒体服务器 Flask04 完整例程cvFlask04 项目的文件树cvFlask04.pyindex1.html 本系列从零…

零基础web安全入门学习路线

相信很多新手都会遇到以下几个问题 1.零基础想学渗透怎么入手&#xff1f; 2.学习web渗透需要从哪里开始&#xff1f; 这让很多同学都处于迷茫状态而迟迟不下手&#xff0c;小编就在此贴给大家说一下web渗透的学习路线&#xff0c;希望对大家有帮助 同时本博客也会按照学习路…

【Java EE 初阶】网络编程套接字TCP的实现

目录 1.实现一个TCP的回显服务 1.Sever Socket API 1.SeverSocket 构造方法 2.Sever Socket方法 2.Socket API 1.Socket的构造方法 2.Socket 方法 那么怎么实现让服务器可以处理多个客户端呢&#xff1f; 服务端代码&#xff1a; 客户端代码&#xff1a; 1.实现一个TC…

【Python】玩转lambda表达式

知识目录 一、写在前面✨二、lambda匿名函数三、泛化函数四、总结撒花&#x1f60a; 一、写在前面✨ 大家好&#xff01;我是初心&#xff0c;又见面了&#xff01; 今天跟大家分享的文章是 玩转Python中的lambda表达式 &#xff0c;希望能帮助到大家&#xff01;本篇文章收录…

三十三、数学知识——质数(朴素筛法 + 埃氏筛法 + 线性筛法)

质数与质数筛法算法主要内容 一、基本思路1、质数质数的判定——试除法&#xff08;复杂度固定位 O(sqrt(n)) &#xff09; 2、分解质因数——试除法&#xff08;最坏是O(sqrt(n))&#xff09;3、朴素筛法——筛的是倍数4、埃氏筛法——朴素筛法优化5、线性筛法——n&#xff0…

刷题---C语言

目录 前言&#xff1a; 一.刷题&#xff08;1&#xff09; 1.1打印X图案 1.2打印带空格直角三角形图案 1.3小乐乐改数字 1.4牛牛的线段 2.刷题&#xff08;2&#xff09; 2.1判断奇偶性 2.2及格分数 2.3kiki算术 2.4&#xff08;ab-c&#xff09;*d 2.5KiKi算期末成…

亿级大表拆分过程记录

两年前接手公司的财务系统的开发和维护工作。在系统移交的初期&#xff0c;笔者和团队就发现&#xff0c;系统内有一张5000W的大表。 跟踪代码发现&#xff0c;该表是用于存储资金流水的表格&#xff0c;关联着众多功能点&#xff0c;同时也有众多的下游系统在使用这张表的数据…

Doris-----Aggregate 聚合模型及案例实现

Aggregate 模型 是相同key的数据进行自动聚合的表模型。表中的列按照是否设置了 AggregationType&#xff0c;分为 Key&#xff08;维度列&#xff09;和 Value&#xff08;指标列&#xff09;&#xff0c;没有设置 AggregationType 的称为 Key&#xff0c;设置了 Aggregation…

外包实在是太坑了,干了三年,感觉人都废了

先说一下自己的情况&#xff0c;专科生&#xff0c;19年通过校招进入杭州某个外包软件公司&#xff0c;干了接近3年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了3年的功…

【JMeter中的View Result Tree显示中文乱码】

JMeter中的View Result Tree显示中文乱码 检查JMeter的安装目录下的bin文件夹中的jmeter.properties配置文件 用记事本打开并搜索&#xff1a;sampleresult.default.encoding 找到该行 改成sampleresult.default.encodingutf-8 修改后重启JMeter ok, 解决乱码 附加 : 下载 J…

【Linux高级 I/O(7)】初识文件锁——fcntl()方法及其独占性、共享性实验(附全文代码)

fcntl()函数在前面系列内容中已经多次用到了&#xff0c;它是一个多功能文件描述符管理工具箱&#xff0c;通过配合不同的 cmd 操作命令来实现不同的功能。为了方便述说&#xff0c;这里再重申一次&#xff1a; #include <unistd.h> #include <fcntl.h>int fcntl(…

大模型对世界的改变,从一时一地,到无处不在、无时不有

作者 | 曾响铃 文 | 响铃说 大模型正在中国遍地开花&#xff0c;做过的没做过的都要过来参合一下。 汹涌浪潮中&#xff0c;不免有更多人开始关注那个最先发布的文心一言。 全球科技大厂中第一个发布GPT大模型产品的百度&#xff0c;在刚刚的中关村论坛上透露了一些文心一言…

nodejs连接mysql

npm i express #node后端框架npm i corsnpm i mysqlconst app require(express)(); const cors require(cors); const port 5000; const mysql require(mysql) //引入mysql 模块app.use(cors({}))const conn mysql.createConnection({user: root,password: qwertyuiop…

普通人想自学软件测试?我还是劝你算了吧。。。

本人7年测试经验&#xff0c;在学测试之前对电脑的认知也就只限于上个网&#xff0c;玩个办公软件。这里不能跑题&#xff0c;我为啥说&#xff1a;自学软件测试&#xff0c;一般人我还是劝你算了吧&#xff1f;因为我就是那个一般人&#xff01; 软件测试基础真的很简单&…

gtest单元测试

gtest单元测试 1. gtest是什么&#xff1f;简答&#xff1a;做测试用的2. gtest的优点3. 搭建测试框架4. gtest_范例演示 1. gtest是什么&#xff1f;简答&#xff1a;做测试用的 gtest是Google的一套用于编写C测试的框架&#xff0c;可以运行在很多平台上&#xff08;包括Lin…

【JavaSE】Java基础语法(十四):Static

文章目录 概述特点与应用注意事项为什么一个静态方法中只能访问用static修饰的成员? 概述 Java中的static是一个修饰符&#xff08;也可称关键字&#xff09;&#xff0c;可以用于修饰变量、方法和代码块。 特点与应用 static修饰的成员具有以下特点&#xff1a; 被类的所有对…

如何在Mac上抓取安卓设备的日志

要在 Mac 上抓取 Android 设备的日志&#xff0c;您可以使用 Android SDK 中的 adb 工具。以下是一个简单的步骤&#xff1a; 1.您需要在 Mac 上安装 Android SDK。您可以从 Android 开发者网站上下载最新版本的 Android SDK&#xff0c;并按照说明进行安装。 2.将您的 Andro…