golang web 开发 —— gin 框架 (gorm 链接 mysql)

2024-04-08 11:28

本文主要是介绍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(&param)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)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/885457

相关文章

MySQL中删除重复数据SQL的三种写法

《MySQL中删除重复数据SQL的三种写法》:本文主要介绍MySQL中删除重复数据SQL的三种写法,文中通过代码示例讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下... 目录方法一:使用 left join + 子查询删除重复数据(推荐)方法二:创建临时表(需分多步执行,逻辑清晰,但会

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

Golang使用minio替代文件系统的实战教程

《Golang使用minio替代文件系统的实战教程》本文讨论项目开发中直接文件系统的限制或不足,接着介绍Minio对象存储的优势,同时给出Golang的实际示例代码,包括初始化客户端、读取minio对... 目录文件系统 vs Minio文件系统不足:对象存储:miniogolang连接Minio配置Min

Golang使用etcd构建分布式锁的示例分享

《Golang使用etcd构建分布式锁的示例分享》在本教程中,我们将学习如何使用Go和etcd构建分布式锁系统,分布式锁系统对于管理对分布式系统中共享资源的并发访问至关重要,它有助于维护一致性,防止竞... 目录引言环境准备新建Go项目实现加锁和解锁功能测试分布式锁重构实现失败重试总结引言我们将使用Go作

mysql重置root密码的完整步骤(适用于5.7和8.0)

《mysql重置root密码的完整步骤(适用于5.7和8.0)》:本文主要介绍mysql重置root密码的完整步骤,文中描述了如何停止MySQL服务、以管理员身份打开命令行、替换配置文件路径、修改... 目录第一步:先停止mysql服务,一定要停止!方式一:通过命令行关闭mysql服务方式二:通过服务项关闭

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

SQL Server数据库磁盘满了的解决办法

《SQLServer数据库磁盘满了的解决办法》系统再正常运行,我还在操作中,突然发现接口报错,后续所有接口都报错了,一查日志发现说是数据库磁盘满了,所以本文记录了SQLServer数据库磁盘满了的解... 目录问题解决方法删除数据库日志设置数据库日志大小问题今http://www.chinasem.cn天发

mysql主从及遇到的问题解决

《mysql主从及遇到的问题解决》本文详细介绍了如何使用Docker配置MySQL主从复制,首先创建了两个文件夹并分别配置了`my.cnf`文件,通过执行脚本启动容器并配置好主从关系,文中还提到了一些... 目录mysql主从及遇到问题解决遇到的问题说明总结mysql主从及遇到问题解决1.基于mysql

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo