Gin框架学习笔记(五) ——文件上传与路由中间件

简介: Gin框架学习笔记(五) ——文件上传与路由中间件

文件上传

单文件上传

这里我准备了一张图片,我们想要实现当我们上传了一个图片的时候,这张图片可以出现在服务器中我们指定的位置


我们可以基于Gin框架来实现我们上述的操作

package main
import "C"
import (
  "fmt"
  "github.com/gin-gonic/gin"
)
func main() {
  r := gin.Default()
  r.MaxMultipartMemory = 8 << 20 // 8 MiB,
  r.POST("/upload", func(c *gin.Context) {
    file, _ := c.FormFile("file")
    fmt.Println(file.Filename)
    dst := "./Source/" + file.Filename
    c.SaveUploadedFile(file, dst)
  })
  r.Run(":8080")
}

编译运行:

保存上传文件

  • SaveUploadedFile函数
    函数签名:
func (c *Context) SaveUploadedFile(file *multipart.FileHeader, dst string)

file:要保存的文件

dst:保存的路径

  • Create and Copy
    这个方法其实也好理解,我们读取上传文件的内容,并且以这份文件的名字创建一个文件,将读到的内容储存到这个文件中,代码如下:
package main
import "C"
import (
  "fmt"
  "github.com/gin-gonic/gin"
  "io"
  "os"
)
func main() {
  r := gin.Default()
  r.MaxMultipartMemory = 8 << 20 // 8 MiB,
  r.POST("/upload", func(c *gin.Context) {
    file, _ := c.FormFile("file")
    fmt.Println(file.Filename)
    fileRead, _ := os.Open(file.Filename)
    defer fileRead.Close()
    dst := "./Source/" + file.Filename
    fileWrite, _ := os.Create(dst)
    defer fileWrite.Close()
    io.Copy(fileWrite, fileRead)
  })
  r.Run(":8080")
}

多文件上传

func file2(c *gin.Context) { //处理多文件上传
  form, _ := c.MultipartForm() //获取表单数据
  files := form.File["upload[]"]
  for _, file := range files {
    fmt.Println(file.Filename)
    dst := "./Source/" + file.Filename
    c.SaveUploadedFile(file, dst)
  }
  c.String(http.StatusOK, fmt.Sprintf("%d files uploaded!", len(files)))
}

路由中间件

前言

我们在日常开发的时候可能忽悠很多公共的业务逻辑,比如说:登录验证,权限校验,数据分页,记录日志等等,我们在每一个模块下都进行相关逻辑的书写无疑是不现实的,在Gin框架中它允许我们在

处理请求的同时自主加入自己的钩子(hook)函数来处理这些公共的业务逻辑,而这些钩子函数也就是我们所说的中间件。

单独注册路由中间件

示例

下面我们首先来看一个很简单的例子:

package main
import "github.com/gin-gonic/gin"
type UserInfo struct {
  Username string `json:"username"`
  Age      int    `json:"age"`
}
func main() {
  r := gin.Default()
  r.GET("/", func(c *gin.Context) {
    user := UserInfo{
      Username: "张三",
      Age:      18,
    }
    c.JSON(200, user)
  })
  r.Run(":8080")
}

运行并向其发出请求:

如上所示,这里的闭包函数就是一个典型的Hook函数,也就是我们所说的中间件。

注意:中间件的类型必须是gin.HandlerFunc类型

单独注册中间件的细节

  • 多个中间件的使用
    我们来看一下GET这一类函数的函数签名:
func (group *RouterGroup) POST(relativePath string, handlers ...HandlerFunc) IRoutes {
  return group.handle(http.MethodPost, relativePath, handlers)
}
// GET is a shortcut for router.Handle("GET", path, handlers).
func (group *RouterGroup) GET(relativePath string, handlers ...HandlerFunc) IRoutes {
  return group.handle(http.MethodGet, relativePath, handlers)
}
// DELETE is a shortcut for router.Handle("DELETE", path, handlers).
func (group *RouterGroup) DELETE(relativePath string, handlers ...HandlerFunc) IRoutes {
  return group.handle(http.MethodDelete, relativePath, handlers)
}
// PATCH is a shortcut for router.Handle("PATCH", path, handlers).
func (group *RouterGroup) PATCH(relativePath string, handlers ...HandlerFunc) IRoutes {
  return group.handle(http.MethodPatch, relativePath, handlers)
}

我们可以发现这里第二个参数的类型为:...HandlerFunc,这意味着我们可以使用在一个请求处理中使用多个中间件函数,比如下面这样:

package main
import "github.com/gin-gonic/gin"
type UserInfo struct {
  Username string `json:"username"`
  Age      int    `json:"age"`
}
func m1(c *gin.Context) {
  user := UserInfo{
    Username: "张三",
    Age:      18,
  }
  c.JSON(200, user)
}
func m2(c *gin.Context) {
  user := UserInfo{
    Username: "李四",
    Age:      20,
  }
  c.JSON(200, user)
}
func main() {
  r := gin.Default()
  r.GET("/", m1, m2)
  r.Run(":8080")
}

运行结果如下:

  • 请求中间件与响应中间件
    一个中间件实现的功能主要是有两部分:对接收到的请求进行相关处理处理即将发送回客户端的响应,而我们根据实现的功能不同其实也可以将一个中间件分为两部分:请求中间件响应中间件,而我们也有专门函数去控制它们的工作,比如下面这个例子:
package main
import (
  "fmt"
  "github.com/gin-gonic/gin"
)
func m1(c *gin.Context) {
  fmt.Println("m1 in")
  c.Next()
  fmt.Println("m1 out")
}
func m2(c *gin.Context) {
  fmt.Println("m2 in")
  c.Next()
  fmt.Println("m2 out")
}
func m3(c *gin.Context) {
  fmt.Println("m3 in")
  c.Next()
  fmt.Println("m3 out")
}
func main() {
  r := gin.Default()
  r.GET("/user", m1, m2, m3)
  r.Run(":8080")
}

运行上面的代码,它的结果是这样的:

m1 in
m2 in
m3 in
m3 out
m2 out
m1 out

我们可以知道它的过程他该是这样的:

这就是Next函数的作用了,它先完成请求中间件的工作部分,再完成响应中间件的工作部分,有趣的是它的模型有点类似于栈的先进后出,大家记忆的时候也可以仿照这个来记忆。

除此之外,我们还可以用Abort函数来拦截响应,示例如下:

package main
import (
  "fmt"
  "github.com/gin-gonic/gin"
)
func m1(c *gin.Context) {
  fmt.Println("m1 in")
  c.Abort()
  fmt.Println("m1 out")
}
func m2(c *gin.Context) {
  fmt.Println("m2 in")
  c.Next()
  fmt.Println("m2 out")
}
func m3(c *gin.Context) {
  fmt.Println("m3 in")
  c.Next()
  fmt.Println("m3 out")
}
func main() {
  r := gin.Default()
  r.GET("/user", m1, m2, m3)
  r.Run(":8080")
}

运行结果为:

m1 in
m1 out

我们可以看到Abort()拦截,后续的HandlerFunc就没有执行了

全局注册中间件

package main
import "github.com/gin-gonic/gin"
func m10(c *gin.Context) {
  c.JSON(200, gin.H{
    "message": "m1",
  })
}
func main() {
  r := gin.Default()
  r.Use(m10) //注册全局中间件
  r.GET("/", func(c *gin.Context) {
    c.JSON(200, gin.H{
      "message": "hello",
    })
  })
  r.Run(":8080")
}

运行结果为:

我们可以发现,虽然我们没有调用m1,但是还是出现了m1的Json响应数据,而这也就是全局注册中间件的作用。

基于中间件传递数据

在Gin框架中,我们可以使用SetGet来实现对数据的设置与接收,代码如下:

package main
import (
  "fmt"
  "github.com/gin-gonic/gin"
)
func m11(c *gin.Context) {
  fmt.Println("m11")
  c.Set("name", "fengxu")
}
func main() {
  r := gin.Default()
  r.Use(m10,m11)
  r.GET("/", func(c *gin.Context) {
    name, _ := c.Get("name")
    c.JSON(200, gin.H{
      "name": name,
    })
  })
  r.Run(":8080")
}

运行结果如下:

细节

这里Get函数的签名如下:

func (c *Context) Set(key string, value any)

所以我们传入任意类型的值,这里我们演示如何传入结构体并获取结构体内指定字段的值:

package main
import (
  "fmt"
  "github.com/gin-gonic/gin"
)
type UserInfo struct {
  Username string `json:"username"`
  Age      int    `json:"age"`
  Sex      string `json:"sex"`
}
func m12(c *gin.Context) {
  user := UserInfo{
    Username: "luoyu",
    Age:      18,
    Sex:      "男",
  }
  c.Set("user", user)
}
func main() {
  r := gin.Default()
  r.Use(m12)
  r.GET("/", m11, m12, func(c *gin.Context) {
    _user, _ := c.Get("user")
    user := _user.(UserInfo)
    c.JSON(200, gin.H{
      "name": user.Username,
    })
  })
  r.Run(":8080")
}

运行结果:

路由分组

基于路由分组的路由管理

在我们日常开发时,可能我们会有很多个路由中间件,如何有效的管理它们就成了一个很大的问题,而这里我们就介绍的是如何基于路由分组来有效的管理路由,首先我们来看一下示例代码:

package main
import "github.com/gin-gonic/gin"
func main() {
  r := gin.Default()
  api := r.Group("api") //设置总的路由
  api.GET("/user", func(c *gin.Context) {
    c.JSON(200, gin.H{
      "msg": "user",
    })
  })
  api.GET("/admin", func(c *gin.Context) {
    c.JSON(200, gin.H{
      "msg": "admin",
    })
  })
  r.Run(":8080")
}

我们可以利用这种分组的方式来有效的实现对路由的组织,防止由于路由过多造成不必要的混乱。

路由分组中间件

路由分组中间件的注册

在Gin框架中,可以去为路由分组去专门订制中间件,以下面的代码为例:

package main
import "github.com/gin-gonic/gin"
type Res struct {
  Code int    `json:"code"`
  Data any    `json:"data"`
  Msg  string `json:"msg"`
}
func m11(c *gin.Context) {
  c.Next()
  c.JSON(200, Res{
    Code: 200,
    Data: "abc",
    Msg:  "响应成功",
  })
}
func main() {
  r := gin.Default()
  api := r.Group("api")
  login := api.Group("login").Use(m11)
  login.GET("/", func(c *gin.Context) {
    c.JSON(200, "ok")
  })
  r.Run(":8080")
}

运行并测试:

我们可以看到这个中间件只有login这个子分组才可以使用这个中间件,我们就实现了一个很简单的路由分组中间件。

相关文章
|
13天前
|
中间件 数据处理 Apache
|
21天前
|
消息中间件 中间件 Kafka
中间件事件总线路由与分发
【6月更文挑战第20天】
17 1
中间件事件总线路由与分发
|
26天前
|
JSON 监控 中间件
中间件在API路由控制
【6月更文挑战第16天】
24 7
|
2月前
|
消息中间件 存储 负载均衡
消息中间件的选择:RabbitMQ是一个明智的选择
消息中间件的选择:RabbitMQ是一个明智的选择
66 0
|
1月前
|
消息中间件 存储 中间件
【消息中间件】详解三大MQ:RabbitMQ、RocketMQ、Kafka
【消息中间件】详解三大MQ:RabbitMQ、RocketMQ、Kafka
82 0
|
12天前
|
消息中间件 编解码 Docker
Docker部署RabbitMQ消息中间件
【7月更文挑战第4天】Docker部署RabbitMQ消息中间件
46 3
|
4天前
|
消息中间件 监控 负载均衡
中间件RabbitMQ性能瓶颈
【7月更文挑战第13天】
37 11
|
15天前
|
消息中间件 NoSQL Kafka
消息中间件(RocketMQ、RabbitMQ、ActiveMQ、Redis、kafka、ZeroMQ)以及之间的区别
消息中间件(RocketMQ、RabbitMQ、ActiveMQ、Redis、kafka、ZeroMQ)以及之间的区别
|
17天前
|
消息中间件 Java Spring
实现Spring Boot与RabbitMQ消息中间件的无缝集成
实现Spring Boot与RabbitMQ消息中间件的无缝集成
|
2月前
|
消息中间件 Java 数据安全/隐私保护
Spring Cloud 项目中实现推送消息到 RabbitMQ 消息中间件
Spring Cloud 项目中实现推送消息到 RabbitMQ 消息中间件