go gin学习记录3

news2025/1/3 3:02:57

环境

环境:mac m1,go version 1.17.2, goland, mysql

安装gorm

第二节学习了在gin中使用go的原生SQL进行操作,这节学习一下使用orm。
go的orm包有很多,gorm是使用较多较广的,所以我们就用gorm来学习。

首先要安装gorm以及相关的驱动:

$ go get -u gorm.io/gorm
go get: added github.com/jinzhu/inflection v1.0.0
go get: added github.com/jinzhu/now v1.1.5
go get: added gorm.io/gorm v1.24.5
$ go get -u gorm.io/driver/mysql
go get: added gorm.io/driver/mysql v1.4.6

安装好gorm之后,我们准备一下要使用的测试数据表,就叫student好了。

建测试表

mysql> create table student(
    -> id int not null auto_increment,
    -> name varchar(32) not null default "" comment "student name",
    -> birth varchar(32) not null default "" comment "student birth",
    -> primary key(id)
    -> )engine=innodb comment "student info";

好的,需要准备的东西都准备好了,下面开始撸码。

搭建代码架子

类似第二节,我们将对student的具体操作都放在一处,还是在controller/目录下新建student.go文件。

package controller

import "github.com/gin-gonic/gin"

type StudentController struct {
}

type Student struct {
	Id    int    `json:"id"`
	Name  string `json:"name"`
	Birth string `json:"birth"`
}

func (s *StudentController) CreateStudent(c *gin.Context) {

}

func (s *StudentController) GetStudentInfo(c *gin.Context) {

}

func (s *StudentController) UpdateStudent(c *gin.Context) {

}

然后在入口main.go中增加一个对student操作的路由组:

	student := r.Group("/student")
	{
		stuCtrl := controller.StudentController{}
		student.POST("/createStudent", stuCtrl.CreateStudent)
		student.GET("/getStudentInfo", stuCtrl.GetStudentInfo)
		student.POST("/updateStudent", stuCtrl.UpdateStudent)
	}

完善插入操作

下面开始完善具体实现。
由于多个方法中都需要使用dsn,所以先把dsn作为一个全局变量放外边定义

const dsn = "root:@tcp(127.0.0.1:3306)/t_gin?charset=utf8mb4&parseTime=True"

完善代码,从插入功能开始。

func (s *StudentController) CreateStudent(c *gin.Context) {
	var stu Student
	stu.Name = c.PostForm("name")
	stu.Birth = c.PostForm("birth")

	db, err := gorm.Open(mysql.Open(dsn))
	if err != nil {
		log.Panic(err.Error())
	}

	result := db.Create(stu)
	if result.Error != nil {
		log.Panic(result.Error.Error())
	}

	c.JSON(http.StatusOK, gin.H{
		"code": 0,
		"data": result.RowsAffected,
	})
}

好的,启动项目,调用一下看看结果
在这里插入图片描述
出错了。
看一下终端的信息:

Error 1146 (42S02): Table 't_gin.students' doesn't exist

这意思是说t_gin数据库里没有找到students表。
这当然找不到,我们建的表明是student。。。所以这是把我们传入的student参数当成要操作的表名了?后面又给加了个s所以成了students?
为了验证这个猜想,我们换个struct当参数试一下

type Haha struct {
	Id    int    `json:"id"`
	Name  string `json:"name"`
	Birth string `json:"birth"`
}

复制一份student,就称之为Haha吧。然后把插入操作的代码调整一下

func (s *StudentController) CreateStudent(c *gin.Context) {
	/*var stu Student
	stu.Name = c.PostForm("name")
	stu.Birth = c.PostForm("birth")*/
	var haha Haha
	haha.Name = c.PostForm("name")
	haha.Birth = c.PostForm("birth")

	db, err := gorm.Open(mysql.Open(dsn))
	if err != nil {
		log.Panic(err.Error())
	}

	result := db.Create(haha)
	if result.Error != nil {
		log.Panic(result.Error.Error())
	}

	c.JSON(http.StatusOK, gin.H{
		"code": 0,
		"data": result.RowsAffected,
	})
}

我们用修改后的代码再重新运行一下,调用看看结果如何:

Error 1146 (42S02): Table 't_gin.hahas' doesn't exist

果然没有意外,如果不知道表名,gorm就按照传入的参数结构体的名称+s作为表名。
有些难以理解。。。
那怎么指定表名呢?
有两种方法,一种是在SQL执行时指定,一种是整个文件范围内的默认指定。
我们先看看第一种,执行时指定:

	result := db.Table("student").Create(haha)

将db操作代码修改如上后,再运行看一下

reflect: reflect.Value.SetInt using unaddressable value

还是报错误,参考了一下官方文档,发现是参数传错了,传参应该给地址,再修改一下:

	result := db.Table("student").Create(&stu)

然后重新执行,这次成功了。
在这里插入图片描述
我们看下数据库里的数据

mysql> select * from student;
+----+------+----------+
| id | name | birth    |
+----+------+----------+
|  5 | toms | 2002-5-1 |
+----+------+----------+
1 row in set (0.01 sec)

下面我们再看一下第二种指定表名的方法,文件中全局指定:
我们修改一下student.go文件,加入下列代码:

type Tabler interface {
	TableName() string
}

func (Student) TableName() string {
	return "student"
}

相当于定义了一个接口并实现了其对应的方法,那我们看看这样是否有效。
继续修改插入操作的代码

	result := db.Create(&stu)

将sql操作中的表名指定去掉,然后我们再运行看看
在这里插入图片描述
运行成功,数据库也有了对应的数据。说明这种方法也是可行的。

完善查询操作

好的,基本的插入操作完善好了,下面来完善查询操作

func (s *StudentController) GetStudentInfo(c *gin.Context) {
	name := c.Query("name")

	db, err := gorm.Open(mysql.Open(dsn))
	if err != nil {
		log.Panic(err.Error())
	}

	var stu Student
	result := db.Where("name=?", name).First(&stu)
	if result.Error == gorm.ErrRecordNotFound {
		c.JSON(http.StatusOK, gin.H{
			"code": 0,
			"data": "",
		})
	}
	if result.Error != nil {
		log.Panic(result.Error.Error())
	}

	c.JSON(http.StatusOK, gin.H{
		"code": 0,
		"data": stu,
	})
}

查询操作完善之后,我们运行一下看看
在这里插入图片描述
非常好,一次成功。
查询是日常开发中最常使用的功能,有许多中查询的方式,以及各种限制条件,这些官方文档里都有详细的说明,想了解具体的就去查一下官方文档,这里不做赘述。

完善更新操作

下面继续完善更新操作,完善后的代码如下:

func (s *StudentController) UpdateStudent(c *gin.Context) {
	name := c.PostForm("name")
	birth := c.PostForm("birth")
	
	db, err := gorm.Open(mysql.Open(dsn))
	if err != nil{
		log.Panic(err.Error())
	}
	
	var stu Student
	result := db.Model(&stu).Where("name=?", name).Update("birth", birth)
	if result.Error != nil {
		log.Panic(result.Error.Error())
	}
	
	c.JSON(http.StatusOK, gin.H{
		"code":0,
		"data": result.RowsAffected,
	})
}

我们继续允许看看
在这里插入图片描述
好的,运行成功了
那数据库中的数据确实更新了吗?

mysql> select * from student;
+----+------+------------+
| id | name | birth      |
+----+------+------------+
|  5 | toms | 2002-5-1   |
|  6 | gaga | 2023-02-17 |
+----+------+------------+
2 rows in set (0.00 sec)

好的,确实更新了,那基本的更新操作也完成了。

获取生成的SQL语句

有些情况下,我们可能想知道生成的sql是什么样子的,原生sql一目了然,而orm却有些云里雾里,就连记录sql log都有些无从下手。
没关系,我们找办法把gorm生成的sql打印出来。
我们用最近的更新操作来做实验,修改更新操作的db操作代码如下:

	var stu Student
	result := db.Model(&stu).Where("name=?", name).Update("birth", birth)
	ss := db.Session(&gorm.Session{DryRun: true}).Model(&stu).Where("name=?", name).Update("birth", birth).Statement
	log.Println(ss.SQL.String(), ss.Vars)

	if result.Error != nil {
		log.Panic(result.Error.Error())
	}

然后我们允许一下看看,就能看到在Terminal终端中打印出了生成的SQL语句:

[GIN-debug] Listening and serving HTTP on :8080
2023/02/17 23:09:56 UPDATE `student` SET `birth`=? WHERE name=? [2000-1-1 gaga]
[GIN] 2023/02/17 - 23:09:56 | 200 |   45.498834ms |             ::1 | POST     "/student/updateStudent"

那那那,gorm生成的sql也打印出来了,排查问题又简单了一步。

好了,今天就到这儿。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/353940.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

车辆逆行识别检测预警算法 yolov5

车辆逆行识别检测预警算法通过Pythonyolov5网络模型计算机算法技术,车辆逆行识别检测预警算法对道路来往行驶车辆出现逆行行为及时预警存档。Python是一种由Guido van Rossum开发的通用编程语言,它很快就变得非常流行,主要是因为它的简单性和…

大数据之-Nifi-监控nifi数据流信息_监控数据来源_bub轻松复现---大数据之Nifi工作笔记0011

通过数据流功能可以轻松复现,数据的流向在某个时间点数据是怎么流动的,出现了什么问题,太强大了.. 真的是,可以看到通过右键,处理器,打开view data province就可以看到, 上面是处理器处理数据的详细信息 点击左侧的详情图标可以查看详情信息,details是这个事件处理的内容详情,…

【计算机网络】运输层

文章目录运输层概述运输层端口号、复用与分用的概念UDP和TCP的对比TCP的流量控制TCP的拥塞控制TCP超时重传时间的选择TCP可靠传输的实现TCP的运输连接管理TCP的连接建立(3次握手)TCP的连接释放(4次挥手)TCP报文段的首部格式运输层概述 这里我们对运输层进行概述,之…

【双指针问题】LeetCode 925. 长按键入

Halo,这里是Ppeua。平时主要更新C语言,C,数据结构算法......感兴趣就关注我吧!你定不会失望。 🌈个人主页:主页链接 🌈算法专栏:专栏链接 我会一直往里填充内容哒! &…

【C++】类型转化

🌈欢迎来到C专栏~~类型转化 (꒪ꇴ꒪(꒪ꇴ꒪ )🐣,我是Scort目前状态:大三非科班啃C中🌍博客主页:张小姐的猫~江湖背景快上车🚘,握好方向盘跟我有一起打天下嘞!送给自己的一句鸡汤&…

Python-第九天 Python异常、模块与包

Python-第九天 Python异常、模块与包一、了解异常1. 什么是异常:2. bug是什么意思:二、异常的捕获方法1. 为什么要捕获异常?2. 捕获异常的语法3. 如何捕获所有异常?三、异常的传递性1.异常是具有传递性的四、Python模块1. 什么是模…

21. 合并两个有序链表

题目链接:解题思路:遍历,双指针:因为两个链表有序,所以只需要依次比较两个元素的大小,然后添加到新的链表中即可first指针指向第一个链表l1,second指针指向第二个链表l2,answer保存合…

Python3 JSON 数据解析

Python3 JSON 数据解析 JSON (JavaScript Object Notation) 是一种轻量级的数据交换格式。 Python3 中可以使用 json 模块来对 JSON 数据进行编解码,它包含了两个函数: json.dumps(): 对数据进行编码。json.loads(): 对数据进行解码。 在 json 的编解码…

CleanMyMac4.12.5最新版安装下载教程

告别硬盘空间不足,让您的Mac极速如新CleanMyMac是一款强大的 Mac 清理、加速工具和健康卫士,让您的 Mac 加快启动速度。CleanMyMac是一款专业的Mac清理软件,可智能清理mac磁盘垃圾和多余语言安装包,快速释放电脑内存,轻…

01:入门篇 - 初识 CTK

作者: 一去、二三里 个人微信号: iwaleon 微信公众号: 高效程序员 CTK 是什么 CTK:支持生物医学图像计算的公共开发包 CTK 全称:Common ToolkitCTK 主页:http://www.commontk.org/Github 地址:https://github.com/commontkCTK 标志 Logo 是一个品牌的形象,对外它传递的…

关于 Docke r安装 Redis 的评论区问题总结及解答

前言: 文章链接:史上最详细Docker安装Redis (含每一步的图解)实战 原文标题只是想让我这篇文章,能够多得到大家的一些关注,事实证明它做到了,当然看到收藏量的那一刻,我也明白&…

数据结构课程设计—简单行编辑程序

简单行编辑程序 一、引言 1.1 问题的提出 文本编辑程序是利用计算机进行文字加工的基本软件工具,实现对文本文件的插入、删除等修改操作。限制这些操作以行为单位进行的编辑程序称为行编辑程序。 被编辑的文本文件可能很大,全部读入编辑程序的数据空间(内存)的作法既不经济,…

强化学习基础知识

强化学习是一种机器学习方法,通过agent与environment的互动,学习适当的action policy以取得更大的奖励reward。本篇博客介绍强化学习的基础知识,与两类强化学习模型。 目录强化学习的基础设定policy based 强化学习的目标3个注意事项实际训练…

Python:跳蚱蜢(BFS判重)

题目描述 本题为填空题,只需要算出结果后,在代码中使用输出语句将所填结果输出即可。 如下图所示: 有 9 只盘子,排成 1 个圆圈。 其中 8 只盘子内装着 8 只蚱蜢,有一个是空盘。 我们把这些蚱蜢顺时针编号为 1 ~ 8。 …

Java【数据结构】—— 冒泡排序、选择排序、直接插入排序

Java实现冒泡排序、选择排序、直接插入排序一、排序的概念及稳定性二、冒泡排序1.基本思想2.代码实现三、选择排序1.基本思想2.代码实现四、直接插入排序1.基本思想2.代码实现五、复杂度及稳定性分析1.冒泡排序2.选择排序3.直接插入排序一、排序的概念及稳定性 - 概念&#xf…

文件断点续传

1、前端上传文件前请求媒资接口层检查文件是否存在,如果已经存在则不再上传。 2、如果文件在系统不存在前端开始上传,首先对视频文件进行分块 3、前端分块进行上传,上传前首先检查分块是否上传,如已上传则不再上传,如…

奇舞周刊第482期:每天都在用,也没整明白的 React Hook

记得点击文章末尾的“ 阅读原文 ”查看哟~下面先一起看下本期周刊 摘要 吧~奇舞推荐■ ■ ■每天都在用,也没整明白的 React Hook推荐理由:针对我们经常使用的 React Hook,本文详细介绍了 useState、useEffect、useContext、useCallback、use…

IP协议格式、IP地址管理、路由选择

目录 一、IP协议格式 16位总长度 ip协议针对传输层的UCP协议或者TCP协议,进行传输的时候,需不需要进行分片传输(拆包传输) 如何进行分片 和 组合分片 8位生存时间: 8位协议、16位校验和、32位源端口、32位目的端…

Agilent E4982A、Keysight E4982A、LCR 表,1 MHz 至 3 GHz

Agilent E4982A、Keysight E4982A、HP E4982A LCR 表,1 MHz 至 3 GHz 产品概览 KEYSIGHT E4982A(安捷伦) Keysight E4982A LCR 表为需要高频(1 MHz 至 3 GHz)阻抗测试的无源元件制造行业提供一流的性能&#xff0c…

Redis实战11-实现优惠券秒杀下单

本篇,咱们来实现优惠券秒杀下单功能。通过本篇学习,我们将会有如下收获: 1:优惠券领券业务逻辑; 2:分析在高并发情况下,出现超卖问题产生的原因; 3:解决超卖问题两种方…