跳至主要內容

Gin

程序员李某某大约 10 分钟

Gin

Hello Gin

go get -u github.com/gin-gonic/gin

package main

import "github.com/gin-gonic/gin"

func main() {

	r := gin.Default()
	r.GET("/helloGin", _helloGin)
    r.Run()	// 空参默认0.0.0.0:8080
    //r.Run(":8080")
    //r.Run("127.0.0.1:8080")
}

func _helloGin(context *gin.Context) {
	context.String(200, "Hello Gin")
}
// 启动方式一
router.Run(":8000")
// 启动方式二,原生方式,需要引入 net/http 包,router.Run本质就是http.ListenAndServe的进一步封装
http.ListenAndServe(":8000", router)

路由

// 默认路由使用了2个中间件Logger(), Recovery()
r := gin.Default()
// 路由分组, 类似java中类上的@RequestMapping
g := r.Group("/sku")
{
    g.GET("/list",_skuList)
    g.POST("/save",_skuSave)
}

响应结果

package main

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

func main() {

	r := gin.Default()

	// 返回页面前要先加载模板文件
	r.LoadHTMLGlob("templates/*")
	// 访问单个文件
	r.StaticFile("/hello", "static/hello.txt") // http://localhost:8080/hello
	// 可通过路径使整个文件夹都可被访问
	r.StaticFS("/static", http.Dir("static")) // http://localhost:8080/static/db.txt

	r.GET("/json", _json)
	r.GET("/helloGin", _helloGin)
	r.GET("/xml", _xml)
	r.GET("/yaml", _yaml)
	r.GET("/index", _index)       // 响应页面
	r.GET("/redirect", _redirect) // 重定向
	r.Run()
}

// 重定向
func _redirect(context *gin.Context) {
	context.Redirect(http.StatusFound, "https://www.baidu.com")
}

// 响应字符串
func _helloGin(context *gin.Context) {
	context.String(http.StatusOK, "Hello Gin")
}

// 响应JSON
func _json(context *gin.Context) {

	type User struct {
		Name string
		Age  int
	}
	user := User{Name: "liyuanhao", Age: 18}

	// 方式一
	context.JSON(http.StatusOK, user)

	// 方式二
	context.JSON(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"mag":    "成功",
		"data": gin.H{
			"name": "lalala",
			"age":  17,
		},
	})
}

// 响应页面
func _index(context *gin.Context) {
	context.HTML(http.StatusOK, "index.html", gin.H{
		"title": "首页",
		"h1":    "你好",
	})
}

// 响应xml
func _xml(context *gin.Context) {
	context.XML(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"mag":    "成功",
		"data": gin.H{
			"name": "lalala",
			"age":  17,
		},
	})
}

// 响应yaml
func _yaml(context *gin.Context) {
	context.YAML(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"mag":    "成功",
		"data": gin.H{
			"name": "lalala",
			"age":  17,
		},
	})
}

请求参数

package main

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

func main() {
	r := gin.Default()
	r.GET("/query", _query)                 // 路径拼参
	r.GET("/param/:userId/:bookId", _param) // restful
	r.POST("/form", _form)                  // 表单参数 multipart/form-data 和 application/x-www-form-urlencoded
	r.POST("/raw", _raw)                    // 原始参数 GetRawData
	r.GET("/header", _header)               // 请求头
	r.GET("/setRespHeader", _setRespHeader) // 设置响应头
	r.Run()
}

func _setRespHeader(context *gin.Context) {
	context.Header("token", "jdalkdjalkdjalkjd")
	context.JSON(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"mag":    "success",
		"data":   "设置响应头",
	})
}

func _header(context *gin.Context) {
	fmt.Println(context.GetHeader("uSer-agent")) // 不区分大小写
	fmt.Println("----------------------------")
	fmt.Println(context.Request.Header)
	fmt.Println("----------------------------")
	fmt.Println(context.Request.Header.Get("user-aGent")) // 不区分大小写
	fmt.Println("----------------------------")
	fmt.Println(context.Request.Header["User-Agent"]) // *** 区分大小写
}

func _raw(context *gin.Context) {
	body, err := context.GetRawData()
	fmt.Println(string(body))
	if err != nil {
		return
	}

	type User struct {
		Name string
		Age  int
	}
	user := User{}
	err = json.Unmarshal(body, &user)
	if err != nil {
		return
	}
	fmt.Println(user)
}

func _form(context *gin.Context) {
	fmt.Println(context.PostForm("user"))
	fmt.Println(context.PostFormArray("user"))
	fmt.Println(context.DefaultPostForm("user", "lalalalalalalla"))
	fmt.Println(context.MultipartForm())
}

func _param(context *gin.Context) {
	fmt.Println(context.Param("bookId"))
	fmt.Println(context.Param("userId"))
}

func _query(context *gin.Context) {
	fmt.Println(context.Query("user"))                     // 存在返回v, 不存在返回“”
	fmt.Println(context.GetQuery("user"))                  // 存在返回v,true, 不存在返回“”,false
	fmt.Println(context.QueryArray("user"))                // 用于返回1k多v的切片
	fmt.Println(context.DefaultQuery("user", "liyuanhao")) // 不存在给个默认值
}

参数绑定

gin中的bind可以很方便的将 前端传递 来的数据与 结构体 进行 参数绑定 ,以及参数校验

在使用这个功能的时候,需要给结构体加上Tag json form uri xml yaml

Must Bind ---- 不用,校验失败会改状态码

Should Bind ---- 可以绑定json,query,param,yaml,xml,如果校验不通过会返回错误

package main

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

func main() {
	r := gin.Default()
	r.POST("/bindJson", _bindJson)               // 绑定json
	r.POST("/bindQuery", _bindQuery)             // 绑定表单
	r.POST("/bindUri/:name/:age/:sex", _bindUri) // 绑定uri
	r.POST("/bind", _bind)                       // 根据 content-type 自动绑定
	r.Run()
}

func _bind(context *gin.Context) {
	user := User{}
	err := context.ShouldBind(&user)
	if err != nil {
		fmt.Println(err)
		context.JSON(200, gin.H{"msg": "你错了"})
		return
	}
	context.JSON(http.StatusOK, user)
}

func _bindUri(context *gin.Context) {
	user := User{}
	err := context.ShouldBindUri(&user)
	if err != nil {
		return
	}
	context.JSON(http.StatusOK, user)
}

func _bindQuery(context *gin.Context) {
	user := User{}
	err := context.ShouldBindQuery(&user)
	if err != nil {
		return
	}
	context.JSON(http.StatusOK, user)
}

type User struct {
	Name string `json:"name" form:"name" uri:"name"`
	Age  int    `json:"age" form:"age" uri:"age"`
	Sex  string `json:"sex" form:"sex" uri:"sex"`
}

func _bindJson(context *gin.Context) {
	user := User{}
	err := context.ShouldBindJSON(&user)
	if err != nil {
		return
	}
	context.JSON(http.StatusOK, user)
}

参数校验

// 不能为空,并且不能没有这个字段
required: 必填字段,如:binding:"required"  

// 针对字符串的长度
min 最小长度,如:binding:"min=5"
max 最大长度,如:binding:"max=10"
len 长度,如:binding:"len=6"

// 针对数字的大小
eq 等于,如:binding:"eq=3"
ne 不等于,如:binding:"ne=12"
gt 大于,如:binding:"gt=10"
gte 大于等于,如:binding:"gte=10"
lt 小于,如:binding:"lt=10"
lte 小于等于,如:binding:"lte=10"

// 针对同级字段的
eqfield 等于其他字段的值,如:PassWord string `binding:"eqfield=Password"`
nefield 不等于其他字段的值


- 忽略字段,如:binding:"-"
// 枚举  只能是red 或green
oneof=red green 

// 字符串  
contains=fengfeng  // 包含fengfeng的字符串
excludes // 不包含
startswith  // 字符串前缀
endswith  // 字符串后缀

// 数组
dive  // dive后面的验证就是针对数组中的每一个元素

// 网络验证
ip
ipv4
ipv6
uri
url
// uri 在于I(Identifier)是统一资源标示符,可以唯一标识一个资源。
// url 在于Locater,是统一资源定位符,提供找到该资源的确切路径

// 日期验证  1月2号下午3点4分5秒在2006年
datetime=2006-01-02

1、结构体通过tag进行校验配置,通过binding绑定

type User struct {
	Name    string `json:"name" binding:"required" msg:"用户名不能为空"`
	Age     int    `json:"age" binding:"gte=18" msg:"年龄必须大于18"`
	Sex     string `json:"sex" binding:"oneof=男 女" msg:"只能填男或女"`
	Address string `json:"address" binding:"max=10" msg:"最大长度10"`
}

2、通过反射,获取tag中的msg,err是ShouldBind的错误信息,obj是绑定的结构体指针,validator这个包要引用v10这个版本的

func GetValidMag(err error, obj any) string {
	getObj := reflect.TypeOf(obj)

    // 强转为校验错误,拿字段名
	if errors, ok := err.(validator.ValidationErrors); ok {
		for _, e := range errors {
			if f, exits := getObj.Elem().FieldByName(e.Field()); exits {
                // 拿字段的mag tag,这里我们只取一个
				return f.Tag.Get("msg")
			}
		}
	}
	return err.Error()
}

3、统一返回结果

type Resp struct {
	Status int    `json:"status"`
	Msg    string `json:"msg"`
	Data   any    `json:"data"`
}

4、handle函数

func _binding(context *gin.Context) {
	user := User{}
	err := context.ShouldBind(&user)
	var msg string
	if err != nil {
		fmt.Println(err)
		msg = GetValidMag(err, &user)
		context.JSON(http.StatusInternalServerError, Resp{
			Status: http.StatusInternalServerError,
			Msg:    msg,
		})
		return
	}
	context.JSON(http.StatusOK, Resp{
		Status: http.StatusOK,
		Msg:    msg,
		Data:   user,
	})
}

自定义校验器

// 注册
func main() {
	r := gin.Default()
	r.POST("/binding", _binding)

	// 自定义校验器
	if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
		// 注册校验器
		v.RegisterValidation("nonBlank", nonBlank)
	}
	r.Run()
}

// 定义
func nonBlank(fl validator.FieldLevel) bool {
	name := fl.Field().Interface().(string)
	if name == "" {
		return false
	}
	return true
}

// 使用
type User struct {
	Name    string `json:"name" binding:"required" msg:"用户名不能为空"`
	Age     int    `json:"age" binding:"gte=18" msg:"年龄必须大于18"`
	Sex     string `json:"sex" binding:"oneof=男 女" msg:"只能填男或女"`
	Address string `json:"address" binding:"nonBlank" msg:"地址不能为空"`
}

文件上传下载

上传

package main

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

func main() {

	r := gin.Default()
	// 为 multipart forms 设置较低的内存限制 (默认是 32 MiB)
	r.MaxMultipartMemory = 8 << 20 // 8 * 2^10 * 2^10
	r.POST("/upload", _upload)
	r.POST("/manyUpload", _manyUpload)
	r.Run()
}

func _manyUpload(context *gin.Context) {
	form, err := context.MultipartForm()
	if err != nil {
		return
	}
	files := form.File["adada"]
	for _, file := range files {
		log.Println(file.Filename)
		// ----------- 以标准流输出 ----------
		open, err := file.Open()
		if err != nil {
			return
		}
		all, err := io.ReadAll(open)
		if err != nil {
			return
		}
		fmt.Println(all)
	}
	context.JSON(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"msg":    fmt.Sprintf("%d files uploaded!", len(files)),
	})
}

func _upload(context *gin.Context) {
	file, err := context.FormFile("adada")
	if err != nil {
		return
	}
	log.Println(file.Filename)
	dst := "./" + file.Filename
	context.SaveUploadedFile(file, dst)

	context.JSON(http.StatusOK, gin.H{
		"status": http.StatusOK,
		"msg":    fmt.Sprintf("%s uploaded!", file.Filename),
	})
}

下载

// 有些响应,比如图片,浏览器就会显示这个图片,而不是下载,所以我们需要使浏览器唤起下载行为
// 表示是文件流,唤起浏览器下载,一般设置了这个,就要设置文件名
c.Header("Content-Type", "application/octet-stream")
// 用来指定下载下来的文件名
c.Header("Content-Disposition", "attachment; filename="+"牛逼.png")
// 表示传输过程中的编码形式,乱码问题可能就是因为它
c.Header("Content-Transfer-Encoding", "binary")


// 直接响应一个路径下的文件
c.File("uploads/12.png")

前后端分离

c.Header("fileName", "xxx.png")
c.Header("msg", "文件下载成功")
c.File("uploads/12.png")

会话

cookie基本不用了

  • 明文,不安全
  • 增加带宽消耗
  • 可以禁用
  • 有上限
func testCookie(c *gin.Context){
   // 获取客户端是否携带cookie
   cookie, err := c.Cookie("key_cookie")
   if err != nil {
      cookie = "NotSet"
      // 给客户端设置cookie
      //  maxAge int, 单位为秒
      // path,cookie所在目录
      // domain string,域名
      //   secure 是否智能通过https访问
      // httpOnly bool  是否允许别人通过js获取自己的cookie
      c.SetCookie("key_cookie", "value_cookie", 60, "/",
         "localhost", false, true)
   }
   fmt.Printf("cookie的值是: %s\n", cookie)
}

session

go get -u github.com/gorilla/sessions

gorilla/sessions为自定义session后端提供cookie和文件系统session以及基础结构。

主要功能是:

  • 简单的API:将其用作设置签名(以及可选的加密)cookie的简便方法。
  • 内置的后端可将session存储在cookie或文件系统中。
  • Flash消息:一直持续读取的session值。
  • 切换session持久性(又称“记住我”)和设置其他属性的便捷方法。
  • 旋转身份验证和加密密钥的机制。
  • 每个请求有多个session,即使使用不同的后端也是如此。
  • 自定义session后端的接口和基础结构:可以使用通用API检索并批量保存来自不同商店的session。
1、初始化store
var store = sessions.NewCookieStore([]byte("something-very-secret"))

2、获取session对象
session, err := store.Get(context.Request, "SESSION")

3、设置
session.Values["user"] = "123"

4、获取
session.Values["user"]

5、删除
session.Options.MaxAge = -1

6、生效
session.Save(context.Request, context.Writer)
package main

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

// 初始化一个cookie存储对象 something-very-secret ---- 加密密钥
var store = sessions.NewCookieStore([]byte("something-very-secret"))

func main() {
	r := gin.Default()
	r.GET("/userInfo", _verify, _userInfo)	// 验证 session
	r.GET("/login", _login)		// 登录 --- 添加 session
	r.GET("/quit", _quit)		// 退出 --- 删除 session
	r.Run()
}

func _quit(context *gin.Context) {
	session, err := store.Get(context.Request, "SESSION")
	if err != nil {
		context.JSON(http.StatusInternalServerError, gin.H{"mag": err})
		return
	}
	// 将session的最大存储时间设置为小于零的数即为删除
	session.Options.MaxAge = -1
	session.Save(context.Request, context.Writer)
	context.JSON(http.StatusOK, gin.H{"msg": "success"})
}

func _login(context *gin.Context) {
	session, err := store.Get(context.Request, "SESSION")
	if err != nil {
		context.JSON(http.StatusInternalServerError, gin.H{"mag": err})
		return
	}
	session.Values["user"] = "123"
	session.Save(context.Request, context.Writer)
	context.JSON(http.StatusOK, gin.H{"msg": "success"})
}

func _verify(context *gin.Context) {
	session, err := store.Get(context.Request, "SESSION")
	fmt.Println(session)

	if err != nil {
		fmt.Println(err)
		context.JSON(500, gin.H{"msg": "请登录"})
		context.Abort()
	} else if session.Values["user"] == "123" {
		context.Next()
	} else {
		context.JSON(500, gin.H{"msg": "第一个中间件拦截了"})
		context.Abort()
	}
}

type User struct {
	Name string
	Age  int
}

func _userInfo(context *gin.Context) {
	user := User{Name: "lll", Age: 11}
	context.JSON(http.StatusOK, user)
}

中间件

类似java的拦截器和Servlet

Gin框架允许开发者在处理请求的过程中,加入用户自己的钩子(Hook)函数。这个钩子函数就叫中间件,中间件适合处理一些公共的业务逻辑,比如登录认证、权限校验、数据分页、记录日志、耗时统计等 即比如,如果访问一个网页的话,不管访问什么路径都需要进行登录,此时就需要为所有路径的处理函数进行统一一个中间件

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

每个路由,路径后面跟的参数其实都是中间件

func (group *RouterGroup) POST(relativePath string, handlers ...HandlerFunc) IRoutes {
	return group.handle(http.MethodPost, relativePath, handlers)
}
r.GET("/login", _login)                // _login 就是一个中间件
r.GET("/userInfo", _verify, _userInfo) // _verify前置可以拦截
func _verify(context *gin.Context) {
	session, err := store.Get(context.Request, "SESSION")
	fmt.Println(session)

	if err != nil {
		fmt.Println(err)
		context.JSON(500, gin.H{"msg": "请登录"})
		context.Abort()								// 拦截
	} else if session.Values["user"] == "123" {
		context.Next()								// 放行
	} else {
		context.JSON(500, gin.H{"msg": "第一个中间件拦截了"})
		context.Abort()
	}
}

注意

context.Next()之前的代码相当于java中的preHandle,之后的代码相当于postHandle

如果其中一个中间件响应了c.Abort(),后续中间件将不再执行,直接按照顺序走完所有的响应中间件

全局中间件

func main() {
	r := gin.Default()
    r.Use(_verify)		// 
	r.GET("/userInfo", _userInfo)
	r.Run()
}
func _verify(context *gin.Context) {
	session, err := store.Get(context.Request, "SESSION")
	fmt.Println(session)

	if err != nil {
		fmt.Println(err)
		context.JSON(500, gin.H{"msg": "请登录"})
		context.Abort()
	} else if session.Values["user"] == "123" {
		context.Next()
	} else {
		context.JSON(500, gin.H{"msg": "第一个中间件拦截了"})
		context.Abort()
	}
}

中间件数据传递

func main() {
	r := gin.Default()
	r.GET("/testSet", _set, _testSet)
	r.Run()
}

// 传递数据
func _set(context *gin.Context) {
	log.Println("### 来到了中间件~~")
	context.Set("name", "lalalal")
}
// 接收数据
func _testSet(c *gin.Context) {
	name, _ := c.Get("name")
	log.Println("拿到了中间件中的name", name)
	c.JSON(http.StatusOK, gin.H{"name": name})
}

gin.Default

从源码中可以看出,默认使用了Logger,Recovery两个中间件

func Default() *Engine {
	debugPrintWARNINGDefault()
	engine := New()
	engine.Use(Logger(), Recovery())
	return engine
}

jwt权限验证

func main() {
	r := gin.Default()
	xxx := r.Group("/xxx")
	api := xxx.Group("/api").Use(_jwt)
	{
		api.GET("/testJWT", _index)
	}
	r.Run()

}

func _jwt(context *gin.Context) {
	token := context.GetHeader("token")
	if token == "1234" {
		// 验证通过
		context.Next()
		return
	}
	context.JSON(http.StatusOK, gin.H{"mag": "请登录"})
	context.Abort()
}

func _index(context *gin.Context) {
	context.JSON(http.StatusOK, gin.H{"msg": "index"})
}

耗时统计

func main() {
	r := gin.Default()
	all := r.Group("").Use(_ConsumeTime)
	{
		all.GET("/ConsumeTime", _login)
	}
	r.Run()
}

func _login(context *gin.Context) {
	context.JSON(http.StatusOK, gin.H{"msg": "yes"})
}

func _ConsumeTime(context *gin.Context) {
	start := time.Now()
	context.Next()
	since := time.Since(start)
	// 获取当前请求所对应的函数
	f := context.HandlerName()
	fmt.Printf("函数 %s 耗时 %d\n", f, since)
}

日志

gin自带的日志

// 输出到文件和控制台
func main() {
	fLog, _ := os.Create("./gin_log.log")				// 创建日志文件
	gin.DefaultWriter = io.MultiWriter(fLog, os.Stdout)	// 配置,输出到文件 且 控制台
    // 设置输出的格式  默认
	r := gin.Default()
	r.GET("/testLog", func(context *gin.Context) {
		context.JSON(http.StatusOK, gin.H{"msg": "yes"})
	})
	r.Run()
}
上次编辑于:
贡献者: ext.liyuanhao3