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数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

基于Python开发Windows自动更新控制工具

《基于Python开发Windows自动更新控制工具》在当今数字化时代,操作系统更新已成为计算机维护的重要组成部分,本文介绍一款基于Python和PyQt5的Windows自动更新控制工具,有需要的可... 目录设计原理与技术实现系统架构概述数学建模工具界面完整代码实现技术深度分析多层级控制理论服务层控制注

mysql8.0.43使用InnoDB Cluster配置主从复制

《mysql8.0.43使用InnoDBCluster配置主从复制》本文主要介绍了mysql8.0.43使用InnoDBCluster配置主从复制,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录1、配置Hosts解析(所有服务器都要执行)2、安装mysql shell(所有服务器都要执行)3、

k8s中实现mysql主备过程详解

《k8s中实现mysql主备过程详解》文章讲解了在K8s中使用StatefulSet部署MySQL主备架构,包含NFS安装、storageClass配置、MySQL部署及同步检查步骤,确保主备数据一致... 目录一、k8s中实现mysql主备1.1 环境信息1.2 部署nfs-provisioner1.2.

MySQL中VARCHAR和TEXT的区别小结

《MySQL中VARCHAR和TEXT的区别小结》MySQL中VARCHAR和TEXT用于存储字符串,VARCHAR可变长度存储在行内,适合短文本;TEXT存储在溢出页,适合大文本,下面就来具体的了解... 目录一、VARCHAR 和 TEXT 基本介绍1. VARCHAR2. TEXT二、VARCHAR

MySQL中C接口的实现

《MySQL中C接口的实现》本节内容介绍使用C/C++访问数据库,包括对数据库的增删查改操作,主要是学习一些接口的调用,具有一定的参考价值,感兴趣的可以了解一下... 目录准备mysql库使用mysql库编译文件官方API文档对象的创建和关闭链接数据库下达sql指令select语句前言:本节内容介绍使用C/

mybatis直接执行完整sql及踩坑解决

《mybatis直接执行完整sql及踩坑解决》MyBatis可通过select标签执行动态SQL,DQL用ListLinkedHashMap接收结果,DML用int处理,注意防御SQL注入,优先使用#... 目录myBATiFBNZQs直接执行完整sql及踩坑select语句采用count、insert、u

MySQL之搜索引擎使用解读

《MySQL之搜索引擎使用解读》MySQL存储引擎是数据存储和管理的核心组件,不同引擎(如InnoDB、MyISAM)采用不同机制,InnoDB支持事务与行锁,适合高并发场景;MyISAM不支持事务,... 目录mysql的存储引擎是什么MySQL存储引擎的功能MySQL的存储引擎的分类查看存储引擎1.命令