gin框架中的路由
基本路由
- gin框架中采用的路由库是基于httrouter做的
- 地址为:https://github.com/julienschmidt/httprouter
httprouter路由库
点击查看代码
package main
import (
"fmt"
"github.com/julienschmidt/httprouter"
"net/http"
)
func main() {
router := httprouter.New()
router.GET("/", index)
router.POST("/hello/:name/:age", hello)
http.ListenAndServe(":8080", router)
}
func index(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
fmt.Fprintln(w, "index页面")
}
func hello(w http.ResponseWriter, r *http.Request, ps httprouter.Params) {
// 请求地址:http://127.0.0.1:8080/hello/lisi/15
fmt.Println(ps, ps[0], ps[0].Key, ps[0].Value)
// 输出结果: [{name lisi} {age 15}] {name lisi} name lisi
fmt.Fprintf(w,"hello: %s: %s", ps.ByName("name"), ps.ByName("age"))
}
请求地址:http://127.0.0.1:8080/hello/lisi
返回结果:hello: lisi
Restful风格的API
gin支持Restful风格的API
即Representational State Transfer的缩写。直接翻译的意思是"表现层状态转化",是一种互联网应用程序的API设计理念:URL定位资源,用HTTP描述操作
1.获取文章 /blog/getXxx Get blog/Xxx
2.添加 /blog/addXxx POST blog/Xxx
3.修改 /blog/updateXxx PUT blog/Xxx
4.删除 /blog/delXxxx DELETE blog/Xxx
API参数
- 可以通过Context的Param方法来获取API参数
- localhost:8000/xxx/zhangsan
type Hello struct {}
func (h *Hello) Register(router *gin.Engine) {
// 绑定路由规则执行的函数
router.GET("/hello/:score/*action", h.Hello)
}
func (h *Hello) Hello(context *gin.Context) {
// gin.Context里面封装了request和response
// 上下文是 gin 中最重要的部分。例如,它允许我们在中间件之间传递变量、管理流程、验证请求的 JSON 并呈现 JSON 响应。
context.JSON(http.StatusOK, gin.H{
"name": "zhang",
"age": 18,
"username": context.MustGet("username"),
"score": context.Param("score"),
// strings.Trim 截取
"action": strings.Trim(context.Param("action"), "/"),
})
}
URL参数
- URL参数可以通过DefaultQuery()或Query()方法获取
- DefaultQuery()若参数不村则,返回默认值,Query()若不存在,返回空串
- API ? name=zs
type Hello struct {}
func (h *Hello) Register(router *gin.Engine) {
// 绑定路由规则执行的函数
router.GET("/hello", h.Hello)
}
func (h *Hello) Hello(context *gin.Context) {
// gin.Context里面封装了request和response
// 上下文是 gin 中最重要的部分。例如,它允许我们在中间件之间传递变量、管理流程、验证请求的 JSON 并呈现 JSON 响应。
context.JSON(http.StatusOK, gin.H{
"name": context.Query("name"),
"age": context.DefaultQuery("age", "18"),
})
}
表单参数
- 表单传输为post请求,http常见的传输格式为四种:
- application/json
- application/x-www-form-urlencoded
- application/xml
- multipart/form-data
- 表单参数可以通过PostForm()方法获取,该方法默认解析的是x-www-form-urlencoded或from-data格式的参数
func (h *Hello) Hello(context *gin.Context) {
// gin.Context里面封装了request和response
// 上下文是 gin 中最重要的部分。例如,它允许我们在中间件之间传递变量、管理流程、验证请求的 JSON 并呈现 JSON 响应。
context.JSON(http.StatusOK, gin.H{
"name": context.PostForm("name"),
"age": context.DefaultPostForm("age", "18"),
})
}
上传单个文件
- multipart/form-data格式用于文件上传
- gin文件上传与原生的net/http方法类似,不同在于gin把原生的request封装到c.Request中
type Hello struct {}
func (h *Hello) Register(router *gin.Engine) {
// 绑定路由规则执行的函数
router.POST("/file", h.Hello)
}
func (h *Hello) Hello(context *gin.Context) {
// gin.Context里面封装了request和response
// 上下文是 gin 中最重要的部分。例如,它允许我们在中间件之间传递变量、管理流程、验证请求的 JSON 并呈现 JSON 响应。
file, err := context.FormFile("file")
if err != nil {
fmt.Println(err)
context.String(http.StatusInternalServerError, "上传图片失败")
}
context.SaveUploadedFile(file, "./test/" + file.Filename)
context.String(http.StatusOK, file.Filename)
}
上传特定文件
有的用户上传文件需要限制上传文件的类型以及上传文件的大小,但是gin框架暂时没有这些函数(也有可能是我没找到),因此基于原生的函数写法自己写了一个可以限制大小以及文件类型的上传函数
点击查看代码
type Hello struct {}
func (h *Hello) Register(router *gin.Engine) {
// 绑定路由规则执行的函数
router.POST("/file", h.Hello)
}
func (h *Hello) Hello(context *gin.Context) {
file, err := context.FormFile("file")
if err != nil {
fmt.Println(err)
return
}
// 判断文件大小
if file.Size > 1024 * 1024 {
fmt.Println("文件太大了")
return
}
// 判断文件类型
if types := file.Header.Get("Content-Type"); types != "image/jpeg" {
fmt.Println("文件类型不对,不是image/jpeg类型")
return
}
context.SaveUploadedFile(file, "./test/" + file.Filename)
context.String(http.StatusOK, file.Filename)
}
上传多个文件
点击查看代码
type Hello struct {}
func (h *Hello) Register(router *gin.Engine) {
// 绑定路由规则执行的函数
router.POST("/file", h.Hello)
}
func (h *Hello) Hello(context *gin.Context) {
form , err := context.MultipartForm()
if err != nil {
fmt.Println(err)
return
}
// 获取所有图片
files := form.File["files"]
// 遍历所有图片
for _, file := range files {
// 逐个存
if err = context.SaveUploadedFile(file, "./test/" + file.Filename); err != nil {
fmt.Println(err)
return
}
}
context.String(http.StatusOK, fmt.Sprintf("upload ok %d files", len(files)))
}
routes group
routes group是为了管理一些相同的URL
- main.go
点击查看代码
package main
import (
"common_standard_library/controller"
"github.com/gin-gonic/gin"
)
func main() {
// 创建路由
router := gin.Default()
// 注册中间件
router.Use(setUserNameHandler)
// 注册前台路由
front := router.Group("/front")
registerFrontRouter(front)
// 注册后台路由
admin := router.Group("/admin")
registerAdminRouter(admin)
router.Run()
}
// 前台路由
func registerFrontRouter(router *gin.RouterGroup) {
// 注册index路由
new(controller.Index).Register(router)
}
// 后台路由
func registerAdminRouter(router *gin.RouterGroup) {
// 注册hello路由
new(controller.Hello).Register(router)
}
// 中间件设置变量
func setUserNameHandler(context *gin.Context) {
context.Set("username", "马亚南")
}
2. index.go
点击查看代码
package controller
import "github.com/gin-gonic/gin"
type Index struct {}
func (i *Index) Register(router *gin.RouterGroup) {
router.GET("/index", i.index)
}
func (i *Index) index(context *gin.Context) {
context.JSON(200, "OK")
}
- hello.go
点击查看代码
package controller
import (
"fmt"
"github.com/gin-gonic/gin"
"net/http"
)
type Hello struct {}
func (h *Hello) Register(router *gin.RouterGroup) {
// 绑定路由规则执行的函数
router.POST("/file", h.hello)
}
func (h *Hello) hello(context *gin.Context) {
form , err := context.MultipartForm()
if err != nil {
fmt.Println(err)
return
}
// 获取所有图片
files := form.File["files"]
// 遍历所有图片
for _, file := range files {
// 逐个存
if err = context.SaveUploadedFile(file, "./test/" + file.Filename); err != nil {
fmt.Println(err)
return
}
}
context.String(http.StatusOK, fmt.Sprintf("upload ok %d files", len(files)))
}
路由原理
- httproter会将所有路由规则构造一颗前缀树
- 例如有 root and as at cn com
gin框架中的路由的更多相关文章
- gin框架中请求路由组的使用
1. gin框架中可以使用路由组来实现对路由的分类 package main import "github.com/gin-gonic/gin" func main() { rou ...
- gin框架中的路由拆分与注册
基本的路由注册 下面最基础的gin路由注册方式,适用于路由条目比较少的简单项目或者项目demo. package main import ( "net/http" "gi ...
- Gin 框架 - 安装和路由配置
目录 概述 Gin 安装 路由配置 推荐阅读 概述 看下 Gin 框架的官方介绍: Gin 是一个用 Go (Golang) 编写的 web 框架. 它是一个类似于 martini 但拥有更好性能的 ...
- 在gin框架中使用JWT
在gin框架中使用JWT JWT全称JSON Web Token是一种跨域认证解决方案,属于一个开放的标准,它规定了一种Token实现方式,目前多用于前后端分离项目和OAuth2.0业务场景下. 什么 ...
- golang gin框架中实现一个简单的不是特别精确的秒级限流器
起因 看了两篇关于golang中限流器的帖子: Gin 开发实践:如何实现限流中间件 常用限流策略--漏桶与令牌桶介绍 我照着用,居然没效果-- 时间有限没有深究.这实在是一个很简单的功能,我的需求是 ...
- 【解决了一个小问题】gin框架中出现如下错误:"[GIN-debug] [WARNING] Headers were already written. Wanted to override status code 400 with 500"
POST到数据到一条gin框架的接口后,客户端收到400错误,并且返回了业务中返回的"decode json fail". 关键代码是: func report(c *gin.Co ...
- golang gin框架中实现大文件的流式上传
一般来说,通过c.Request.FormFile()获取文件的时候,所有内容都全部读到了内存.如果是个巨大的文件,则可能内存会爆掉:且,有的时候我们需要一边上传一边处理. 以下的代码实现了大文件流式 ...
- gin框架中中间件的编写与使用
概念 一个完整的系统可能包括鉴权认证.权限管理.安全检查.日志记录等多维度的系统支持. 中间件位与服务器和实际业务处理程序之间,其含义就相当于在请求和具体的业务处理逻辑之间增加某些操作,这种以额外增加 ...
- gin框架中的参数验证
结构体验证 用gin框架的数据验证,可以不用解析数据,减少if else,会简洁许多. 处理请求方法 func structValidator(context *gin.Context) { var ...
随机推荐
- 《Java必须知道的300个问题》读书总结
这本书是在图书馆随便逛的时候找到的书.花了一下午看完了,感觉有用的地方不是很多,大部分都是些概念,并没有太大用途.不过里边有些东西还是可以看一看的,总结如下. Java语言基础 1.表达式3-2.6= ...
- 【LeetCode】146. LRU Cache 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 字典+双向链表 日期 题目地址:https://le ...
- 【LeetCode】63. Unique Paths II 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题目地址:https://leetcode.com/problems/unique-pa ...
- 【LeetCode】647. Palindromic Substrings 解题报告(Python)
[LeetCode]647. Palindromic Substrings 解题报告(Python) 标签: LeetCode 题目地址:https://leetcode.com/problems/p ...
- 【LeetCode】860. Lemonade Change 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...
- 【剑指Offer】52. 两个链表的第一个公共节点 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 解题方法 方法一:栈 方法二:HashSet 方法三:不使用额外空间 日期 ...
- mac学习Python第一天:安装、软件说明、运行python的三种方法
一.Python安装 从Python官网下载Python 3.x的安装程序,下载后双击运行并安装即可: Python有两个版本,一个是2.x版,一个是3.x版,这两个版本是不兼容的. MAC 系统一般 ...
- CS5265替代LT8711设计TYPEC转HDMI 4K高清投屏方案|LT8711龙迅替代方案
龙迅LT8711是一款Type-C/DP1.2 to HDMI2.0方案芯片.LT8711HE是一款高性能Type-C/DP1.2至HDMI2.0转换器,设计用于将USB typec或DP1.2源连接 ...
- MySQL百分比显示和显示前百分之几的方法
前几天一个朋友让我帮忙写的,随手记录一下,感觉难度也不大,就是写的时候遇到一些问题.优化方便做得不太好.有好的优化方法欢迎分享!(数据库在文章结尾) 要求 1)查询所有时间内,所有产品销售金额占比,按 ...
- 【MySQL作业】连接查询——美和易思内连接查询应用习题
点击打开所使用到的数据库>>> 1.使用内连接获取客户"王传华"所有的订单信息和客户信息. 使用内连接获取客户"王传华"所有的订单信息和客户信 ...