firstName := c.Query("first\_name")
lastName := c.DefaultQuery("last\_name", "wang") // 默认值
c.String(http.StatusOK, "%s:%s", firstName, lastName)
})
// 启动
r.Run(":8080")
}
###### 获取POST请求参数
package main
import ( "github.com/gin-gonic/gin" "net/http" )
func main() {
// 创建实例
r := gin.Default()
r.POST("/test", func(c \*gin.Context) {
// 从POST请求中获取参数值
firstName := c.PostForm("first\_name")
lastName := c.DefaultPostForm("last\_name", "wei")
c.String(http.StatusOK, "%s:%s", firstName, lastName)
})
// 启动
r.Run(":8080")
}
###### 获取Body值
package main
import ( "bytes" "github.com/gin-gonic/gin" "io/ioutil" "net/http" )
func main() {
// 创建实例
r := gin.Default()
r.POST("/test", func(c \*gin.Context) {
// 数据流中获取Body
all, err := ioutil.ReadAll(c.Request.Body)
if err != nil {
c.String(http.StatusBadRequest, err.Error())
// 结束, 将程序中止
c.Abort()
}
// 从数据流取出body后, 无法再通过post获取到POST请求中参数的值
// 需要将读到的结果再回存, 再写到c.Request.Body中
c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(all))
firstName := c.PostForm("first\_name")
lastName := c.DefaultPostForm("last\_name", "wei")
c.String(http.StatusOK, "%s:%s:%s", firstName, lastName, string(all))
})
// 启动
r.Run(":8080")
}
###### 获取参数绑定到结构体
package main
import ( "github.com/gin-gonic/gin" "net/http" "time" )
// Person 定义结构体
type Person struct {
Name string form:"name"
Address string form:"address"
Birthday time.Time form:"birthday" time\_format:"2006-01-02"
}
func main() {
// 创建实例
r := gin.Default()
r.GET("/test", handle)
r.POST("/test", handle)
r.Run(":8080")
}
func handle(c *gin.Context) {
var user Person
// 这里是根据请求的content-type来做不同的binding操作
err := c.ShouldBind(&user)
// 结构体要传指针, 因为要修改值, 并且结构体字段首字母大写, 因为要包外可见
if err != nil {
c.String(http.StatusBadRequest, "error: %v", err)
c.Abort()
}
c.String(http.StatusOK, "user: %v", user)
}
#### 2. 验证请求参数
###### 结构体验证
更多验证规则[官方验证规则](https://gitee.com/vip204888)
package main
import ( "github.com/gin-gonic/gin" "net/http" )
// Person 定义结构体
type Person struct {
// 验证规则 多个条件都须满足用,分割 多个条件任意满足用|分割, 多个条件时,和|两边不要有空格
Name string form:"name" binding:"required" // 结构体绑定通过form标签, 结构体字段验证通过binding标签
Address string form:"address" binding:"required" // 结构体绑定通过form标签, 结构体字段验证通过binding标签
Age int form:"age" binding:"required,gt=10" // 结构体绑定通过form标签, 结构体字段验证通过binding标签
}
func main() {
// 创建实例
r := gin.Default()
// 请求 http://127.0.0.1:8080/testing?name=chao&age=10&address=hangzhou
// 返回 error: Key: 'Person.Age' Error:Field validation for 'Age' failed on the 'gt' tag
r.GET("/testing", handle)
r.Run(":8080")
}
func handle(c *gin.Context) {
var user Person
// 结构体要传指针, 因为要修改值, 并且结构体字段首字母大写, 因为要包外可见
if err := c.ShouldBind(&user); err != nil {
// 这里是根据请求的content-type来做不同的binding操作
c.String(http.StatusInternalServerError, "error: %v", err)
c.Abort() // Note that this will not stop the current handler, call Abort to ensure the remaining handlers for this request are not called.
return
}
c.String(http.StatusOK, "user: %v", user)
}
###### 自定义验证
package main
import ( "fmt" "github.com/gin-gonic/gin" "github.com/gin-gonic/gin/binding" "github.com/go-playground/validator/v10" "net/http" "time" )
// Booking 定义结构体
type Booking struct {
// 验证规则 多个条件都须满足用,分割 多个条件任意满足用|分割 多个条件时,和|两边不要有空格
CheckIn time.Time form:"check\_in" binding:"required,bookabledate" time\_format:"2006-01-02" // 结构体绑定通过form标签, 结构体字段验证通过binding标签
CheckOut time.Time form:"check\_out" binding:"required,gtfield=CheckIn" time\_format:"2006-01-02" // 结构体绑定通过form标签, 结构体字段验证通过binding标签
}
// 自定义验证规则 func bookableDate(fl validator.FieldLevel) bool {
// 类型断言, 是否 time.Time 类型
if date, ok := fl.Field().Interface().(time.Time); ok {
// 当前时间, 格式为 Time
today := time.Now()
// 转成时间戳格式
if today.Unix() < date.Unix() {
return true
}
}
return false
}
func main() {
// 创建实例
r := gin.Default()
// 注册验证器
if validate, ok := binding.Validator.Engine().(\*validator.Validate); ok {
// 将 tag 中的 key, 与自定义规则方法绑定
err := validate.RegisterValidation("bookabledate", bookableDate)
if err != nil {
fmt.Println(err)
return
}
}
// 请求 http://127.0.0.1:8080/bookable?check\_in=2021-11-04&check\_out=2021-11-05
// 返回 {"error": "Key: 'Booking.CheckIn' Error:Field validation for 'CheckIn' failed on the 'bookabledate' tag"}
// 请求 http://127.0.0.1:8080/bookable?check\_in=2021-11-08&check\_out=2021-11-02
// 返回 {"error": "Key: 'Booking.CheckOut' Error:Field validation for 'CheckOut' failed on the 'gtfield' tag"}
r.GET("/bookable", handle)
r.Run(":8080")
}
func handle(c *gin.Context) {
var booking Booking
// 结构体要传指针, 因为要修改值, 并且结构体字段首字母大写, 因为要包外可见
if err := c.ShouldBind(&booking); err != nil {
// 这里是根据请求的content-type来做不同的binding操作
c.JSON(http.StatusInternalServerError, gin.H{
"error": err.Error(),
})
return
}
c.JSON(http.StatusOK, gin.H{
"message": "ok",
"booking": booking,
})
}
###### 升级验证-支持多语言错误信息
package main
import ( "github.com/gin-gonic/gin" en2 "github.com/go-playground/locales/en" // 英文语言包 zh2 "github.com/go-playground/locales/zh" // 中文语言包 ut "github.com/go-playground/universal-translator" "github.com/go-playground/validator/v10" // 公共包 en_translations "github.com/go-playground/validator/v10/translations/en" zh_translations "github.com/go-playground/validator/v10/translations/zh" "net/http" )
// Person 定义结构体
type Person struct {
// 验证规则 多个条件都须满足用,分割 多个条件任意满足用|分割 多个条件时,和|两边不要有空格
// 使用多语言包, 结构体验证 tag 不再是 binding, 而是 validate
Name string form:"name" validate:"required" json:"name"
Age int form:"age" validate:"required,gt=10" json:"age"
}
func main() {
// 创建验证器
v := validator.New()
// 支持的语言
zh := zh2.New()
en := en2.New()
// 创建翻译器
translator := ut.New(zh, en)
// 创建实例
r := gin.Default()
// 请求 http://127.0.0.1:8080/bookable?name=chao&age=20, 返回 {"data":{"name":"chao","age":20},"message":"ok"}
// 请求 http://127.0.0.1:8080/bookable?age=20, 返回 {"error":["Name为必填字段"]}
// 请求 http://127.0.0.1:8080/bookable?name=chao&age=10&locale=en, 返回 {"error":["Age must be greater than 10"]}
// 请求 http://127.0.0.1:8080/bookable, 返回 {"error":["Name为必填字段","Age为必填字段"]}
r.GET("/bookable", func(c \*gin.Context) {
// 接收指定语言参数, 设置默认语言
locale := c.DefaultQuery("locale", "zh")
trans, \_ := translator.GetTranslator(locale)
switch locale {
case "zh":
zh_translations.RegisterDefaultTranslations(v, trans)
case "en":
en_translations.RegisterDefaultTranslations(v, trans)
default:
zh_translations.RegisterDefaultTranslations(v, trans)
}
var p Person
// 绑定结构体
if err := c.ShouldBind(&p); err != nil {
// 结构体要传指针, 因为要修改值, 并且结构体字段首字母大写, 因为要包外可见
c.JSON(http.StatusInternalServerError, gin.H{
"error": err.Error(),
})
c.Abort()
return
}
// 验证结构体
if err := v.Struct(p); err != nil {
errs := err.(validator.ValidationErrors)
var sliceErrors []string
for \_, e := range errs {
// 将错误翻译成相应的语言
sliceErrors = append(sliceErrors, e.Translate(trans))
}
c.JSON(http.StatusInternalServerError, gin.H{
"error": sliceErrors,
})
c.Abort()
return
}
c.JSON(http.StatusOK, gin.H{
"message": "ok",
"data": p,
})
})
r.Run(":8080")
}
#### 3. 中间件
中间件是介于 gin 服务器和执行的回调函数之间的中间层, 可以作为请求拦截和日志打印
###### 使用 gin 中间件
package main
import ( "github.com/gin-gonic/gin" "io" "net/http" "os" )
func main() {
// 指定日志写入文件
file, \_ := os.Create("gin.log")
gin.DefaultWriter = io.MultiWriter(file)
gin.DefaultErrorWriter = io.MultiWriter(file)
// 创建实例, gin.Default()方法默认已经实现了Logger()和Recovery()两个中间件
r := gin.New()
// 设置中间件
// gin.Logger() 添加日志中间件 打印到终端, 或者写入文件
// gin.Recovery() 添加 panic 恢复中间件, 遇到 panic 不会中断程序, 如果不加该中间件会直接中断程序结束运行
r.Use(gin.Logger(), gin.Recovery())
// [GIN] 2021/11/05 - 22:45:53 | 200 | 997.8µs | 127.0.0.1 | GET "/get"
r.GET("/get", func(c \*gin.Context) {
name := c.DefaultQuery("name", "chao")
c.JSON(http.StatusOK, gin.H{
"message": "OK",
"data": name,
})
})
r.Run(":8080")
}
###### 自定义 ip 白名单中间件
package main
import ( "github.com/gin-gonic/gin" "net/http" )
// 白名单 var ipList = []string{ "127.0.0.2", }
// IPAuthMiddleware 验证IP func IPAuthMiddleware() gin.HandlerFunc {
return func(c \*gin.Context) {
// 获取客户端IP
clientIP := c.ClientIP()
flag := false
for \_, ip := range ipList {
if ip == clientIP {
// 在列表中
flag = true
break
}
}
// 不在白名单中
if !flag {
c.JSON(http.StatusUnauthorized, gin.H{
"message": "no auth",
})
c.Abort()
}
}
} func main() {
// 创建实例
r := gin.Default()
// 设置自定义中间件
r.Use(IPAuthMiddleware())
// 请求 http://127.0.0.1:8080/get
// 返回 {"message":"no auth"}
r.GET("/get", func(c \*gin.Context) {
name := c.DefaultQuery("name", "chao")
c.JSON(http.StatusOK, gin.H{
"message": "OK",
"data": name,
})
})
r.Run(":8080")
}
#### 4. 优雅关停服务器

传统的 gin.Run 是阻塞的, 一直监听端口, 关闭后服务直接结束, 而优雅关闭服务中 server.ListenAndServer是不阻塞的, 用 os.Signal 去阻塞, 当监听到 os.Signal 信号, 将超时的上下文传送到 Shutdown 方法中, 然后退出, 在监听到信号之后到正式退出之前, 会关闭这个时间段内重新进来的连接请求, 另外在超时时间内把之前已经接收到的请求执行完毕
package main
import ( "context" "github.com/gin-gonic/gin" "log" "net/http" "os" "os/signal" "syscall" "time" )
func main() {
// 创建实例
r := gin.Default()
// 优雅关停, 即当服务器关闭时, 不会立即关闭对请求链接的响应, 而是将超时时间内的重新链接关闭, 之前的链接会执行完
r.GET("/test", func(c \*gin.Context) {
time.Sleep(10 \* time.Second)
c.JSON(http.StatusOK, gin.H{
"message": "OK",
})
})
// 创建 http服务
server := &http.Server{
Addr: ":8085",
Handler: r,
}
// 开启协程, 监听服务
go func() {
if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
log.Fatalln("listen error: ", err)
}
}()
// 设置请求拦截, 将信号写入管道
quit := make(chan os.Signal)
// 捕获退出和中止信号, 并写入管道中
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
// 阻塞, 直到通道中有值
<-quit
log.Println("shutdown server")
// 创建超时的上下文, 即监听到关闭的信号后, 设置10秒的超时缓冲段
ctx, cancel := context.WithTimeout(context.Background(), 10 \* time.Second)
defer cancel()
// 真正关闭服务的操作
if err := server.Shutdown(ctx); err != nil {
log.Fatalln("shutdown error: ", err)
}
log.Fatalln("server exiting")
}
#### 5. 模板渲染
package main
import ( "github.com/gin-gonic/gin" "net/http" )
func main() {
// 创建实例
r := gin.Default()
// 载入模板
r.LoadHTMLGlob("template/\*")
// 路由回调
r.GET("/index", func(c \*gin.Context) {
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!