golang web 开发 —— gin 框架 (gorm 链接 mysql)
目录
1. 介绍
2. 环境
3. gin
3.1 gin提供的常见路由
3.2 gin的分组
main.go
router.go
代码结构
3.3 gin 提供的Json方法
main.go
route.go
common.go
user.go
order.go
3.4 gin框架下如何获取传递来的参数
第一种是GET请求后面直接 /拼上传递的参数
第二种是这样传递
第三种是以Json方式来传递
3.5 golang如何进行异常捕获和处理
4. 关于日志
5. gin如何操作数据库(需要用到gorm)
main.go
config
db.go
controller
common.go
order.go
user.go
dao
dao.go
models
user.go
router
router.go
1. 介绍


2. 环境

查看是否设置成功

(PS:go env -w GOSUMDB=sum.golang.org(GOSUM问题))
3. gin
gin 的路由是通过前缀树来实现的,每个请求方法一个前缀树(没有使用反射来实现,比反射的效率好得多),一般路由分为(GET,POST,DELETE等等,我们都来看看)
3.1 gin提供的常见路由
package mainimport ("net/http""github.com/gin-gonic/gin")func main() {//先生成一个实例r:= gin.Default()//再定义一个路由//第一个是访问的url 第二个是处理函数r.GET("/hello", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "Hello, World!")})r.POST("/user/list", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user list")})r.PUT("/user/add", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user add")})r.DELETE("/user/delete", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user delete")})//可以使用ANY方法来处理所有的请求r.Any("/any", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "Hello, World!")})//最后启动web服务//可以指定端口号r.Run(":9999")
}
3.2 gin的分组
并且 gin 的路由是可以分组的(/user/list /user/add这种)
package mainimport ("net/http""github.com/gin-gonic/gin")func main() {//先生成一个实例r:= gin.Default()//再定义一个路由//第一个是访问的url 第二个是处理函数// r.GET("/hello", func(ctx *gin.Context) {// ctx.String(http.StatusOK/*200*/, "Hello, World!")// })// r.POST("/user/list", func(ctx *gin.Context) {// ctx.String(http.StatusOK/*200*/, "user list")// })// r.PUT("/user/add", func(ctx *gin.Context) {// ctx.String(http.StatusOK/*200*/, "user add")// })// r.DELETE("/user/delete", func(ctx *gin.Context) {// ctx.String(http.StatusOK/*200*/, "user delete")// })user:=r.Group("user"){//因为上面已经有/user了 所以这里不用再写/useruser.POST("/list", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user list")})user.PUT("/add", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user add")})user.DELETE("/delete", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user delete")})}//最后启动web服务//可以指定端口号r.Run(":9999")
}
然后此时是写在main包里面的,而main包作为入口文件,把这么多的路由写在这里显然很不合理,所以我们创建一个router包去专门导入路由
main.go
package mainimport (//引入包这个整了很久,一直有各种问题//使用注意:要在 GOROOT/src 同级目录下创建项目,然后进入项目文件夹,//然后 go mod init 该文件夹,然后去使用 "gin-ranking/router"
)func main() {r:=router.Router()r.Run(":9999")
}
router.go
package router//和文件夹名保持一致import ("github.com/gin-gonic/gin""net/http"
)func Router() *gin.Engine {r := gin.Default()user:=r.Group("user"){//因为上面已经有/user了 所以这里不用再写/useruser.POST("/list", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user list")})user.PUT("/add", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user add")})user.DELETE("/delete", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user delete")})}return r
}
代码结构

3.3 gin 提供的Json方法
gin返回Json数据很简单,因为他提供了Json方法
路由以及统一的返回

main.go
package mainimport ("gin-ranking/router"
)func main() {r:=router.Router()r.Run(":9999")
}
route.go
package router//和文件夹名保持一致import ("github.com/gin-gonic/gin""net/http""gin-ranking/controller"
)func Router() *gin.Engine {r := gin.Default()user:=r.Group("user"){//因为上面已经有/user了 所以这里不用再写/user 注意使用Talend的时候,前面是 ip:port/user/各种信息user.GET("/info",controller.UserController{}.GetUserInfo)user.POST("/list", controller.UserController{}.GetList)user.PUT("/add", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user add")})user.DELETE("/delete", func(ctx *gin.Context) {ctx.String(http.StatusOK/*200*/, "user delete")})}order:=r.Group("/order"){order.POST("/list", controller.OrderController{}.GetList)}return r
}
common.go
package controllerimport ("github.com/gin-gonic/gin"
)type JsonStruct struct {Code int `json:"code"` //返回的是 200 400 404 500...Msg interface{} `json:"msg"` //返回的信息Data interface{} `json:"data"` //返回的数据Count int64 `json:"count"` //返回的数据总数
}type JsonErrorStruct struct {Code int `json:"code"` //返回的是 200 400 404 500...Msg interface{} `json:"msg"` //返回的信息
}func ReturnSucces(c *gin.Context, code int,msg interface{}, data interface{}, count int64) {json := &JsonStruct{Code: code,Msg: msg,Data: data,Count: count,}c.JSON(200, json)
}func ReturnError(c *gin.Context, code int, msg interface{}) {json := &JsonErrorStruct{Code: code,Msg: msg,}c.JSON(200, json)
}
user.go
package controllerimport ("github.com/gin-gonic/gin"
)//因为我们目前这么写的话,这些文件都在一个包也就其实是
//一个作用域,所以函数名容易重复,此时我们可以给他们加上一个前缀
//比如这里的UserController
type UserController struct {}func (u UserController)GetUserInfo(c *gin.Context) {ReturnSucces(c,0,"succes","user info",1)
}func (u UserController)GetList(c *gin.Context) {ReturnError(c,4004,"没有相关信息list")
}
order.go
package controllerimport ("github.com/gin-gonic/gin"
)type OrderController struct {}func (o OrderController)GetList(c *gin.Context){ReturnError(c,4004,"没有相关信息order")
}
3.4 gin框架下如何获取传递来的参数
传递参数一般分为三种:
第一种是GET请求后面直接 /拼上传递的参数

第二种是这样传递

第三种是以Json方式来传递
然后我们分别来看如何获取:
一 url方式:
这种我们一般会在路由中直接拼上



第二种是通过POST怎么接收值呢

这种方式路由是不用改的,直接改order.go


第三种是Json格式的数据传入

接收Json数据我们需要通过map或者结构体


对于Json还有一种解决方式就是用结构体

3.5 golang如何进行异常捕获和处理
其他语言底层抛出异常上层逻辑通过try catch捕获异常,但是go语言的设计者认为将异常与控制结构混在一起会很容易使得代码变得混乱,为了不影响代码的原有逻辑,采用的延迟执行捕获异常的模式来实现
defer:压栈延迟执行
recover:是个内建函数,可以让系统崩溃的流程恢复过来,仅在延迟函数defer内有效,在正常程序执行中调用会返回nil,并且没有任何效果。假如有问题调用recover可以捕获到panic的输入值,并且恢复正常的执行。
panic:让程序直接崩溃


我们改一下user.go

然后加上

web访问结果就是200,并且没有内容了

并且打印里面捕获到了异常

自定义log中间件,并实现日志的收集
4. 关于日志
///
关于日志,比较重要的有三种:
第一种是项目的请求日志和ngix类似,每次请求都保留日志
第二种是错误日志,当程序出现错误的时候,日志应该保留下来
第三种是在开发业务逻辑的时候,程序员自己保存的日志。比如请求第三方接口的时候,把接口返回的信息直接保留下来。
///
5. gin如何操作数据库(需要用到gorm)

gorm中文文档:GORM - The fantastic ORM library for Golang, aims to be developer friendly.
项目结构

main.go
package mainimport ("gin-ranking/router"
)func main() {r:=router.Router()r.Run(":9999")
}
config
db.go
package configconst (//Mysqldb = "root:123456@tcp(10.18.11.64:3306)/lml_dtbase?charset=utf8"
)
controller
common.go
package controllerimport ("github.com/gin-gonic/gin"
)type JsonStruct struct {Code int `json:"code"` //返回的是 200 400 404 500...Msg interface{} `json:"msg"` //返回的信息Data interface{} `json:"data"` //返回的数据Count int64 `json:"count"` //返回的数据总数
}type JsonErrorStruct struct {Code int `json:"code"` //返回的是 200 400 404 500...Msg interface{} `json:"msg"` //返回的信息
}func ReturnSucces(c *gin.Context, code int,msg interface{}, data interface{}, count int64) {json := &JsonStruct{Code: code,Msg: msg,Data: data,Count: count,}c.JSON(200, json)
}func ReturnError(c *gin.Context, code int, msg interface{}) {json := &JsonErrorStruct{Code: code,Msg: msg,}c.JSON(200, json)
}
order.go
package controllerimport ("github.com/gin-gonic/gin"
)type OrderController struct{}type Search struct {//需要注意的是定义结构体和返回的时候,//都要指定Json的字段,否则首字母大写肯定是匹配不上的//注意前端传来的数据一定是小写的所以我们统一转成小写Name string `json:"name"`Cid int `json:"cid"`
}func (o OrderController) GetList(c *gin.Context) {//cid:=c.PostForm("cid")//name:=c.DefaultPostForm("name","wangwu(Default)")// param := make(map[string]interface{})// err := c.BindJSON(¶m)search := &Search{}err := c.BindJSON(&search)if err == nil {ReturnSucces(c, 0, search.Name, search.Cid, 1)return}ReturnError(c, 4001, gin.H{"err": err})
}
user.go
package controllerimport ("fmt""gin-ranking/models""strconv"//"gin-ranking/pkg/logger""github.com/gin-gonic/gin"
)// 因为我们目前这么写的话,这些文件都在一个包也就其实是
// 一个作用域,所以函数名容易重复,此时我们可以给他们加上一个前缀
// 比如这里的UserController
type UserController struct{}func (u UserController) GetUserInfo(c *gin.Context) {idStr := c.Param("id")name := c.Param("name")id, _ := strconv.Atoi(idStr) //----------testuser, _ := models.GetUserTest(id) //-----testReturnSucces(c, 0, name, user, 1)
}func (u UserController) AddUser(c *gin.Context) {username := c.DefaultPostForm("username", "")id, err := models.AddUser(username)if err != nil {ReturnError(c, 4002, "保存错误")return}ReturnSucces(c, 0, "保存成功", id, 1)
}func (u UserController) UpdateUser(c *gin.Context) {username := c.DefaultPostForm("username", "") //获取usernameidStr := c.DefaultPostForm("id", "")id, _ := strconv.Atoi(idStr)models.UpdateUser(id, username)ReturnSucces(c, 0, "更新成功", true, 1)
}func (u UserController) DeleteUser(c *gin.Context) {idStr := c.DefaultPostForm("id", "")id, _ := strconv.Atoi(idStr)err := models.DeleteUser(id)if err != nil {ReturnError(c, 4002, "删除错误")return}ReturnSucces(c, 0, "删除成功", true, 1)
}func (u UserController) GetList(c *gin.Context) {//user loggerdefer func() {if err := recover(); err != nil {fmt.Println("异常捕获", err)}}()num1 := 1num2 := 0num3 := num1 / num2ReturnError(c, 4004, num3)//ReturnError(c, 4004, "没有相关信息list")
}func (u UserController) GetUserListTest(c *gin.Context) {users, err := models.GetUserListTest()if err != nil {ReturnError(c, 4004, "没有相关信息list")return}ReturnSucces(c, 0, "获取成功", users, 1)
}
dao
dao.go
package daoimport ("fmt""gin-ranking/config""time""github.com/jinzhu/gorm"_ "github.com/jinzhu/gorm/dialects/mysql"
)// 因为要和数据库建立连接,所以我们要把链接保存到var中
var (Db *gorm.DBerr error
)func init() {Db, err = gorm.Open("mysql", config.Mysqldb)if err != nil {fmt.Println("-----mysql connect error:", err)}if Db.Error != nil {fmt.Println("-----database err:", err)}Db.DB().SetMaxIdleConns(10)Db.DB().SetMaxOpenConns(100)Db.DB().SetConnMaxLifetime(time.Hour)
}
models
user.go
package daoimport ("fmt""gin-ranking/config""time""github.com/jinzhu/gorm"_ "github.com/jinzhu/gorm/dialects/mysql"
)// 因为要和数据库建立连接,所以我们要把链接保存到var中
var (Db *gorm.DBerr error
)func init() {Db, err = gorm.Open("mysql", config.Mysqldb)if err != nil {fmt.Println("-----mysql connect error:", err)}if Db.Error != nil {fmt.Println("-----database err:", err)}Db.DB().SetMaxIdleConns(10)Db.DB().SetMaxOpenConns(100)Db.DB().SetConnMaxLifetime(time.Hour)
}
router
router.go
package daoimport ("fmt""gin-ranking/config""time""github.com/jinzhu/gorm"_ "github.com/jinzhu/gorm/dialects/mysql"
)// 因为要和数据库建立连接,所以我们要把链接保存到var中
var (Db *gorm.DBerr error
)func init() {Db, err = gorm.Open("mysql", config.Mysqldb)if err != nil {fmt.Println("-----mysql connect error:", err)}if Db.Error != nil {fmt.Println("-----database err:", err)}Db.DB().SetMaxIdleConns(10)Db.DB().SetMaxOpenConns(100)Db.DB().SetConnMaxLifetime(time.Hour)
}
本文根据小破站作者 慕课网官方账号 GO流行的Gin框架快速搭建开发 所著
在此感谢该博主讲的很棒!
最后的最后,创作不易,希望读者三连支持 💖
赠人玫瑰,手有余香 💖
相关文章:
golang web 开发 —— gin 框架 (gorm 链接 mysql)
目录 1. 介绍 2. 环境 3. gin 3.1 gin提供的常见路由 3.2 gin的分组 main.go router.go 代码结构 3.3 gin 提供的Json方法 main.go route.go common.go user.go order.go 3.4 gin框架下如何获取传递来的参数 第一种是GET请求后面直接 /拼上传递的参数 第二种是…...
区块链相关概念
区块链是什么,就算是做计算机技术开发的程序员,100个当中都没有几个能把这个概念理解明白,更不要说讲清楚了。那对于普通人来说,就更扯了。 除了“挖矿”表面意思似乎比较好理解外,其他的基础概念真TMD绕。 去中心化、…...
文章解读与仿真程序复现思路——电力系统自动化EI\CSCD\北大核心《考虑灵活爬坡产品的虚拟电厂两阶段分布鲁棒优化运营策略》
本专栏栏目提供文章与程序复现思路,具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…...
2.k8s架构
目录 k8s集群架构 控制平面 kube-apiserver kube-scheduler etcd kube-controller-manager node 组件 kubelet kube-proxy 容器运行时(Container Runtime) cloud-controller-manager 相关概念 k8s集群架构 一个Kubernetes集群至少包含一个控制…...
xss.pwnfunction-Ligma
首先用jsFuckhttps://jsfuck.com/ [][(![][])[[]](![][])[![]![]](![][])[![]](!![][])[[]]][([][(![][])[[]](![][])[![]![]](![][])[![]](!![][])[[]]][])[![]![]![]](!![][][(![][])[[]](![][])[![]![]](![][])[![]](!![][])[[]]])[![][[]]]([][[]][])[![]](![][])[![]![]!…...
分布式限流——Redis实现令牌桶算法
令牌桶算法 令牌桶算法(Token Bucket Algorithm)是一种广泛使用的流量控制(流量整形)和速率限制算法。这个算法能够控制网络数据的传输速率,确保数据传输的平滑性,防止网络拥堵,同时也被应用于…...
鸿蒙原生应用已超4000个!
鸿蒙原生应用已超4000个! 来自 HarmonyOS 微博近期消息,#鸿蒙千帆起# 重大里程碑!目前已有超4000个应用加入鸿蒙生态。从今年1月18日华为宣布首批200多家应用厂商正在加速开发鸿蒙原生应用,到3月底超4000个应用,短短…...
manga-ocr漫画日文ocr
github 下载 解压 anaconda新建环境 conda create -n manga_ocr python3.8 激活环境 conda activate manga_ocr cd到解压目录 cd /d manga-ocr-master 安装依赖包 pip install -r requirements.txt pip3 install manga-ocr 下载离线model huggingface 123云盘 解压到一个目录…...
STL、Vector和Set的讲解和例题分析
STL STL(Standard Template Library,标准模板库)是C标准库的一部分,它提供了一系列通用的编程组件,包括容器、迭代器、算法和函数对象等。STL是C中实现泛型编程的核心,它允许程序员使用模板编写与数…...
Android 13 aosp hiddenapi config
Android 11 hiddenapi路径 frameworks/base/config/hiddenapi-greylist-packages.txtAndroid 13 hiddenapi路径 frameworks/base/boot/hiddenapi/hiddenapi-unsupported-packages.txt...
数据仓库面试总结
文章目录 1.什么是数据仓库?2.ETL是什么?3.数据仓库和数据库的区别(OLTP和OLAP的区别)4.数据仓库和数据集市的区别5.维度分析5.1 什么是维度?5.2什么是指标? 6.什么是数仓建模?7.事实表7.维度表…...
git Failed to connect to 你的网址 port 8282: Timed out
git Failed to connect to 你的网址 port 8282: Timed out 出现这个问题的原因是:原来的仓库换了网址,原版网址不可用了。 解决方法如下: 方法一:查看git用户配置是否有如下配置 http.proxyhttp://xxx https.proxyhttp://xxx如果…...
[C++][算法基础]堆排序(堆)
输入一个长度为 n 的整数数列,从小到大输出前 m 小的数。 输入格式 第一行包含整数 n 和 m。 第二行包含 n 个整数,表示整数数列。 输出格式 共一行,包含 m 个整数,表示整数数列中前 m 小的数。 数据范围 1≤m≤n≤&#x…...
备考ICA----Istio实验15---开启 mTLS 自动双向认证实验
备考ICA----Istio实验15—开启mTLS自动双向认证实验 在某些生成环境下,我们希望微服务和微服务之间使用加密通讯方式来确保不被中间人代理. 默认情况下Istio 使用 PERMISSIVE模式配置目标工作负载,PERMISSIVE模式时,服务可以使用明文通讯.为了只允许双向 TLS 流量,…...
Hive SchemaTool 命令详解
Hive schematool 是 hive 自带的管理 schema 的相关工具。 列出详细说明 schematool -help直接输入 schematool 或者schematool -help 输出结果如下: usage: schemaTool-alterCatalog <arg> Alter a catalog, requires--catalogLocation an…...
51单片机入门_江协科技_17~18_OB记录的笔记
17. 定时器 17.1. 定时器介绍:51单片机的定时器属于单片机的内部资源,其电路的连接和运转均在单片机内部完成,无需占用CPU外围IO接口; 定时器作用: (1)用于计时系统,可实现软件计时&…...
xss.pwnfunction-Ah That‘s Hawt
<svg/onloadalert%26%2340%3B1%26%2341%3B> <svg/>是一个自闭合形式 ,当页面或元素加载完成时,onload 事件会被触发,从而可以执行相应的 JavaScript 函数...
Python学习从0开始——005数据结构
Python学习从0开始——005数据结构 一、列表list二、元组和序列三、集合四、字典五、循环技巧六、条件控制七、序列和其它类型的比较 一、列表list 不是所有数据都可以排序或比较。例如,[None, ‘hello’, 10] 就不可排序,因为整数不能与字符串对比&…...
力扣每日一题:LCR112--矩阵中的最长递增路径
题目 给定一个 m x n 整数矩阵 matrix ,找出其中 最长递增路径 的长度。 对于每个单元格,你可以往上,下,左,右四个方向移动。 不能 在 对角线 方向上移动或移动到 边界外(即不允许环绕)。 示例…...
树莓派部署yolov5实现目标检测(ubuntu22.04.3)
最近两天搞了一下树莓派部署yolov5,有点难搞(这个东西有点老,版本冲突有些包废弃了等等) 最后换到ubuntu系统弄了,下面是我的整体步骤(建议先使能一下ssh(最下面有),结合…...
JavaSec-RCE
简介 RCE(Remote Code Execution),可以分为:命令注入(Command Injection)、代码注入(Code Injection) 代码注入 1.漏洞场景:Groovy代码注入 Groovy是一种基于JVM的动态语言,语法简洁,支持闭包、动态类型和Java互操作性,…...
Python:操作 Excel 折叠
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...
LeetCode - 394. 字符串解码
题目 394. 字符串解码 - 力扣(LeetCode) 思路 使用两个栈:一个存储重复次数,一个存储字符串 遍历输入字符串: 数字处理:遇到数字时,累积计算重复次数左括号处理:保存当前状态&a…...
定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器
第一章 引言:语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域,文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量,支撑着搜索引擎、推荐系统、…...
【JavaSE】绘图与事件入门学习笔记
-Java绘图坐标体系 坐标体系-介绍 坐标原点位于左上角,以像素为单位。 在Java坐标系中,第一个是x坐标,表示当前位置为水平方向,距离坐标原点x个像素;第二个是y坐标,表示当前位置为垂直方向,距离坐标原点y个像素。 坐标体系-像素 …...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
OPENCV形态学基础之二腐蚀
一.腐蚀的原理 (图1) 数学表达式:dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一,腐蚀跟膨胀属于反向操作,膨胀是把图像图像变大,而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...
关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...
