【gin学习笔记】08gorm结构体的创建和结合gin的使用_gorm varchar-程序员宅基地

技术标签: 学习  golang  数据库  gin  

本文学习视频 https://www.bilibili.com/video/BV1kC4y1t7Qi/?spm_id_from=333.788

tag设置

  • 设置主键: gorm:“primary_key”
  • 自定义字段名称: column:user_id
  • 忽略: “-”
  • 指定数据类型t:ype:varchar(100)
  • 非空:not null
  • 创建索引: index
  • 设置外键:ForeignKey
  • 关联外键:AssociationForeignKey
  • 多对多:many2many:表面

自定义表名

一般如果不指定的话,表名就是结构体的名字,如果想自定义表表名要用到TableName()这个方法

tag设置和自定义表名的代码如下:

package main

import (
	_ "github.com/go-sql-driver/mysql"
	"github.com/jinzhu/gorm"
)

type User struct {
    
	gorm.Model
	//设置主键,字段名称设置为user_name,字段类型设置为varchar长度为100
	Name string `gorm:"primary_key;column:user_name;type:varchar(100)"`
	Age  int
}

//将表名设置为my_users
func (u User) TableName() string {
    
	return "my_users"
}

func main() {
    
	db, _ := gorm.Open("mysql",
		"root:root@/ginclass?charset=utf8mb4&parseTime=True&loc=Local")
	defer db.Close()
	db.AutoMigrate(&User{
    })
}

运行后创建出来的表结构如下:
在这里插入图片描述

重点:结构体声明 1对1 1对多 多对多

  • 多对多使用many2many关键字
  • 分页查询使用count记录总数,使用limit和offset指定记录位置
  • 预加载 preload

举例:
一对一 一个学生有一个身份证
一对多 一个学生有一个班级,一个班级有多个学生
多对多 一个学生有多个老师 一个老师有多个学生

下面举例来创建一些表并生成一些数据

package main

import (
	_ "github.com/go-sql-driver/mysql"
	"github.com/jinzhu/gorm"
)

//一对一:学生每个人有一张IDCard,IDCard通过ID来知道是哪个学生的,这是一对一关系
//一对多:班级有多个学生,学生通过ClassID知道自己在哪个班
//多对多:老师有多个学生,并且知道学生的ID;学生有多个老师,并且知道老师的ID

type Class struct {
    
	gorm.Model
	ClassName string
	Students  []Student //一个班级有多个学生
}

type Student struct {
    
	gorm.Model
	StudentName string
	ClassID     uint      //每个学生都有一个班级
	IDCard      IDCard    //每个学生都有一张IDCard
	Teachers    []Teacher `gorm:"many2many:student_teachers;"` //一个学生有多个老师 student_teachers是关联表
}

type Teacher struct {
    
	gorm.Model
	TeacherName string
	StudentID   uint
	Students    []Student `gorm:"many2many:student_teachers;"` //一个老师有多个学生
}

type IDCard struct {
    
	gorm.Model
	StudentID uint //每个学生有一个IDcard 与学生的ID进行关联
	Num       int
}

func main() {
    
	db, _ := gorm.Open("mysql",
		"root:root@/ginclass?charset=utf8mb4&parseTime=True&loc=Local")
	defer db.Close()
	
	//创建Teacher表,Class表,Student表,IDCard表,还有之前的关联表student_teachers
	db.AutoMigrate(&Teacher{
    }, &Class{
    }, &Student{
    }, &IDCard{
    })
	
	i := IDCard{
    
		Num: 123456,
	}
	//创建一个学生
	s := Student{
    
		StudentName: "kaka",
		IDCard:      i,
	}
	//创建一个老师
	t := Teacher{
    
		TeacherName: "张老师",
	}
	//创建一个班级
	c := Class{
    
		ClassName: "文科班",
		Students:  []Student{
    s},
	}
	_ = db.Create(&c).Error //把班级信息写到数据库
	_ = db.Create(&t).Error //把老师信息写到数据库
}

运行后会在数据库中看到如下信息:
在这里插入图片描述
class表中创建了一条名字为文科班的数据,id为1
在这里插入图片描述
学生表中创建了一条学生名字为kaka的数据,班级id为1,学生id为1
在这里插入图片描述
IDCard表中创建了一条卡号为123456的数据
在这里插入图片描述
teacher表中也相应创建了一条数据
但是many2many关联表student_teachers里面没有数据,这里面的数据怎么创建呢?
代码如下,只需要在teacher中加入Students: []Student{s} 即可

//创建一个老师
	t := Teacher{
    
		TeacherName: "张老师",
		Students:    []Student{
    s},
	}

运行后结果如下
在这里插入图片描述
创建了id为2的老师数据

在这里插入图片描述
创建了id为3的student数据
在这里插入图片描述
teacher 2 和student 3对应上了

使用gin接受参数并经过处理入库或返回

我们创建一个/student路由

package main

import (
	"github.com/gin-gonic/gin"
	_ "github.com/go-sql-driver/mysql"
	"github.com/jinzhu/gorm"
)

//一对一:学生每个人有一张IDCard,IDCard通过ID来知道是哪个学生的,这是一对一关系
//一对多:班级有多个学生,学生通过ClassID知道自己在哪个班
//多对多:老师有多个学生,并且知道学生的ID;学生有多个老师,并且知道老师的ID

type Class struct {
    
	gorm.Model
	ClassName string
	Students  []Student //一个班级有多个学生
}

type Student struct {
    
	gorm.Model
	StudentName string
	ClassID     uint      //每个学生都有一个班级
	IDCard      IDCard    //每个学生都有一张IDCard
	Teachers    []Teacher `gorm:"many2many:student_teachers;"` //一个学生有多个老师 student_teachers是关联表
}

type Teacher struct {
    
	gorm.Model
	TeacherName string
	StudentID   uint
	Students    []Student `gorm:"many2many:student_teachers;"` //一个老师有多个学生
}

type IDCard struct {
    
	gorm.Model
	StudentID uint //每个学生有一个IDcard 与学生的ID进行关联
	Num       int
}

func main() {
    
	db, _ := gorm.Open("mysql",
		"ginclass:Fjj@sh123@/ginclass?charset=utf8mb4&parseTime=True&loc=Local")
	defer db.Close()

	r := gin.Default()
	r.POST("/student", func(c *gin.Context) {
    
		var student Student
		_ = c.BindJSON(&student)
		db.Create(&student)
	})
	r.Run(":8888")
}

在这里插入图片描述
post请求的header中加上 Content-Type applicaiton/json,用来告诉服务端消息主体是序列化后的 JSON 字符串。
在这里插入图片描述
Body中传入以上信息,运行后数据正常入库。在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

用gin路由查询数据库里的数据

创建一个get路由

package main

import (
	"github.com/gin-gonic/gin"
	_ "github.com/go-sql-driver/mysql"
	"github.com/jinzhu/gorm"
)

//一对一:学生每个人有一张IDCard,IDCard通过ID来知道是哪个学生的,这是一对一关系
//一对多:班级有多个学生,学生通过ClassID知道自己在哪个班
//多对多:老师有多个学生,并且知道学生的ID;学生有多个老师,并且知道老师的ID

type Class struct {
    
	gorm.Model
	ClassName string
	Students  []Student //一个班级有多个学生
}

type Student struct {
    
	gorm.Model
	StudentName string
	ClassID     uint      //每个学生都有一个班级
	IDCard      IDCard    //每个学生都有一张IDCard
	Teachers    []Teacher `gorm:"many2many:student_teachers;"` //一个学生有多个老师 student_teachers是关联表
}

type Teacher struct {
    
	gorm.Model
	TeacherName string
	StudentID   uint
	Students    []Student `gorm:"many2many:student_teachers;"` //一个老师有多个学生
}

type IDCard struct {
    
	gorm.Model
	StudentID uint //每个学生有一个IDcard 与学生的ID进行关联
	Num       int
}

func main() {
    
	db, _ := gorm.Open("mysql",
		"ginclass:Fjj@sh123@/ginclass?charset=utf8mb4&parseTime=True&loc=Local")
	defer db.Close()

	r := gin.Default()
	r.POST("/student", func(c *gin.Context) {
    
		var student Student
		_ = c.BindJSON(&student)
		db.Create(&student)
	})

	r.GET("/student/:ID", func(c *gin.Context) {
    
		id := c.Param("ID") //从前端拿到ID,并赋给id
		var student Student
		db.First(&student, "id=?", id) //通过id去数据库里查询一条数据并把结果返回给student
		//将student返回前端
		c.JSON(200, gin.H{
    
			"学生信息": student,
		})
	})

	r.Run(":8888")
}

在这里插入图片描述
查询id为2的学生信息,可以看到返回的只有基础信息,其他的信息都没返回。为了将其他信息也查出来就需要preload

db.Preload("Teachers").Preload("IDCard").First(&student, "id=?", id) //通过id去数据库里查询一条数据并把结果返回给student
		//将student返回前端

db.First前面加Preload
重新get下会发现返回的信息变多了,结果如下:

{
    
    "学生信息": {
    
        "ID": 2,
        "CreatedAt": "2022-05-12T23:35:38+08:00",
        "UpdatedAt": "2022-05-12T23:35:38+08:00",
        "DeletedAt": null,
        "StudentName": "Jack",
        "ClassID": 1,
        "IDCard": {
    
            "ID": 4,
            "CreatedAt": "2022-05-12T23:35:38+08:00",
            "UpdatedAt": "2022-05-12T23:35:38+08:00",
            "DeletedAt": null,
            "StudentID": 2,
            "Num": 666666
        },
        "Teachers": [
            {
    
                "ID": 3,
                "CreatedAt": "2022-05-12T23:35:38+08:00",
                "UpdatedAt": "2022-05-12T23:35:38+08:00",
                "DeletedAt": null,
                "TeacherName": "语文老师",
                "StudentID": 0,
                "Students": null
            },
            {
    
                "ID": 4,
                "CreatedAt": "2022-05-12T23:35:38+08:00",
                "UpdatedAt": "2022-05-12T23:35:38+08:00",
                "DeletedAt": null,
                "TeacherName": "英语老师",
                "StudentID": 0,
                "Students": null
            }
        ]
    }
}

接下来通过class把students都查出来

r.GET("/class/:ID", func(c *gin.Context) {
    
		id := c.Param("ID") //从前端拿到ID,并赋给id
		var class Class
		db.First(&class, "id=?", id) //通过id去数据库里查询一条数据并把结果返回给class
		//将class返回前端
		c.JSON(200, gin.H{
    
			"班级信息": class,
		})
	})

查询后返回结果如下
在这里插入图片描述
可以看到没有Students信息,接下来我们通过preload将students信息查询出来。

r.GET("/class/:ID", func(c *gin.Context) {
    
		id := c.Param("ID") //从前端拿到ID,并赋给id
		var class Class
		db.Preload("Students").First(&class, "id=?", id) //通过id去数据库里查询一条数据并把结果返回给class
		//将class返回前端
		c.JSON(200, gin.H{
    
			"班级信息": class,
		})
	})

结果如下

{
    
    "班级信息": {
    
        "ID": 1,
        "CreatedAt": "2022-05-12T22:06:51+08:00",
        "UpdatedAt": "2022-05-12T22:06:51+08:00",
        "DeletedAt": null,
        "ClassName": "文科班",
        "Students": [
            {
    
                "ID": 2,
                "CreatedAt": "2022-05-12T23:35:38+08:00",
                "UpdatedAt": "2022-05-12T23:35:38+08:00",
                "DeletedAt": null,
                "StudentName": "Jack",
                "ClassID": 1,
                "IDCard": {
    
                    "ID": 0,
                    "CreatedAt": "0001-01-01T00:00:00Z",
                    "UpdatedAt": "0001-01-01T00:00:00Z",
                    "DeletedAt": null,
                    "StudentID": 0,
                    "Num": 0
                },
                "Teachers": null
            }
        ]
    }
}

可以看到students信息确实出来了,但是student关联的信息没有查出来。
这个就涉及到嵌套预加载的问题。

r.GET("/class/:ID", func(c *gin.Context) {
    
		id := c.Param("ID") //从前端拿到ID,并赋给id
		var class Class
		db.Preload("Students").Preload("Students.IDCard").Preload("Students.Teachers").First(&class, "id=?", id) //通过id去数据库里查询一条数据并把结果返回给class
		//将class返回前端
		c.JSON(200, gin.H{
    
			"班级信息": class,
		})
	})

嵌套了Students.IDCard, Students.Teachers
返回结果如下

{
    
    "班级信息": {
    
        "ID": 1,
        "CreatedAt": "2022-05-12T22:06:51+08:00",
        "UpdatedAt": "2022-05-12T22:06:51+08:00",
        "DeletedAt": null,
        "ClassName": "文科班",
        "Students": [
            {
    
                "ID": 2,
                "CreatedAt": "2022-05-12T23:35:38+08:00",
                "UpdatedAt": "2022-05-12T23:35:38+08:00",
                "DeletedAt": null,
                "StudentName": "Jack",
                "ClassID": 1,
                "IDCard": {
    
                    "ID": 4,
                    "CreatedAt": "2022-05-12T23:35:38+08:00",
                    "UpdatedAt": "2022-05-12T23:35:38+08:00",
                    "DeletedAt": null,
                    "StudentID": 2,
                    "Num": 666666
                },
                "Teachers": [
                    {
    
                        "ID": 3,
                        "CreatedAt": "2022-05-12T23:35:38+08:00",
                        "UpdatedAt": "2022-05-12T23:35:38+08:00",
                        "DeletedAt": null,
                        "TeacherName": "语文老师",
                        "StudentID": 0,
                        "Students": null
                    },
                    {
    
                        "ID": 4,
                        "CreatedAt": "2022-05-12T23:35:38+08:00",
                        "UpdatedAt": "2022-05-12T23:35:38+08:00",
                        "DeletedAt": null,
                        "TeacherName": "英语老师",
                        "StudentID": 0,
                        "Students": null
                    }
                ]
            }
        ]
    }
}

其他的gorm操作可以参考https://learnku.com/docs/gorm/v2

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/kilmerfun/article/details/124697755

智能推荐

oracle 12c 集群安装后的检查_12c查看crs状态-程序员宅基地

文章浏览阅读1.6k次。安装配置gi、安装数据库软件、dbca建库见下:http://blog.csdn.net/kadwf123/article/details/784299611、检查集群节点及状态:[root@rac2 ~]# olsnodes -srac1 Activerac2 Activerac3 Activerac4 Active[root@rac2 ~]_12c查看crs状态

解决jupyter notebook无法找到虚拟环境的问题_jupyter没有pytorch环境-程序员宅基地

文章浏览阅读1.3w次,点赞45次,收藏99次。我个人用的是anaconda3的一个python集成环境,自带jupyter notebook,但在我打开jupyter notebook界面后,却找不到对应的虚拟环境,原来是jupyter notebook只是通用于下载anaconda时自带的环境,其他环境要想使用必须手动下载一些库:1.首先进入到自己创建的虚拟环境(pytorch是虚拟环境的名字)activate pytorch2.在该环境下下载这个库conda install ipykernelconda install nb__jupyter没有pytorch环境

国内安装scoop的保姆教程_scoop-cn-程序员宅基地

文章浏览阅读5.2k次,点赞19次,收藏28次。选择scoop纯属意外,也是无奈,因为电脑用户被锁了管理员权限,所有exe安装程序都无法安装,只可以用绿色软件,最后被我发现scoop,省去了到处下载XXX绿色版的烦恼,当然scoop里需要管理员权限的软件也跟我无缘了(譬如everything)。推荐添加dorado这个bucket镜像,里面很多中文软件,但是部分国外的软件下载地址在github,可能无法下载。以上两个是官方bucket的国内镜像,所有软件建议优先从这里下载。上面可以看到很多bucket以及软件数。如果官网登陆不了可以试一下以下方式。_scoop-cn

Element ui colorpicker在Vue中的使用_vue el-color-picker-程序员宅基地

文章浏览阅读4.5k次,点赞2次,收藏3次。首先要有一个color-picker组件 <el-color-picker v-model="headcolor"></el-color-picker>在data里面data() { return {headcolor: ’ #278add ’ //这里可以选择一个默认的颜色} }然后在你想要改变颜色的地方用v-bind绑定就好了,例如:这里的:sty..._vue el-color-picker

迅为iTOP-4412精英版之烧写内核移植后的镜像_exynos 4412 刷机-程序员宅基地

文章浏览阅读640次。基于芯片日益增长的问题,所以内核开发者们引入了新的方法,就是在内核中只保留函数,而数据则不包含,由用户(应用程序员)自己把数据按照规定的格式编写,并放在约定的地方,为了不占用过多的内存,还要求数据以根精简的方式编写。boot启动时,传参给内核,告诉内核设备树文件和kernel的位置,内核启动时根据地址去找到设备树文件,再利用专用的编译器去反编译dtb文件,将dtb还原成数据结构,以供驱动的函数去调用。firmware是三星的一个固件的设备信息,因为找不到固件,所以内核启动不成功。_exynos 4412 刷机

Linux系统配置jdk_linux配置jdk-程序员宅基地

文章浏览阅读2w次,点赞24次,收藏42次。Linux系统配置jdkLinux学习教程,Linux入门教程(超详细)_linux配置jdk

随便推点

matlab(4):特殊符号的输入_matlab微米怎么输入-程序员宅基地

文章浏览阅读3.3k次,点赞5次,收藏19次。xlabel('\delta');ylabel('AUC');具体符号的对照表参照下图:_matlab微米怎么输入

C语言程序设计-文件(打开与关闭、顺序、二进制读写)-程序员宅基地

文章浏览阅读119次。顺序读写指的是按照文件中数据的顺序进行读取或写入。对于文本文件,可以使用fgets、fputs、fscanf、fprintf等函数进行顺序读写。在C语言中,对文件的操作通常涉及文件的打开、读写以及关闭。文件的打开使用fopen函数,而关闭则使用fclose函数。在C语言中,可以使用fread和fwrite函数进行二进制读写。‍ Biaoge 于2024-03-09 23:51发布 阅读量:7 ️文章类型:【 C语言程序设计 】在C语言中,用于打开文件的函数是____,用于关闭文件的函数是____。

Touchdesigner自学笔记之三_touchdesigner怎么让一个模型跟着鼠标移动-程序员宅基地

文章浏览阅读3.4k次,点赞2次,收藏13次。跟随鼠标移动的粒子以grid(SOP)为partical(SOP)的资源模板,调整后连接【Geo组合+point spirit(MAT)】,在连接【feedback组合】适当调整。影响粒子动态的节点【metaball(SOP)+force(SOP)】添加mouse in(CHOP)鼠标位置到metaball的坐标,实现鼠标影响。..._touchdesigner怎么让一个模型跟着鼠标移动

【附源码】基于java的校园停车场管理系统的设计与实现61m0e9计算机毕设SSM_基于java技术的停车场管理系统实现与设计-程序员宅基地

文章浏览阅读178次。项目运行环境配置:Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。项目技术:Springboot + mybatis + Maven +mysql5.7或8.0+html+css+js等等组成,B/S模式 + Maven管理等等。环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。_基于java技术的停车场管理系统实现与设计

Android系统播放器MediaPlayer源码分析_android多媒体播放源码分析 时序图-程序员宅基地

文章浏览阅读3.5k次。前言对于MediaPlayer播放器的源码分析内容相对来说比较多,会从Java-&amp;amp;gt;Jni-&amp;amp;gt;C/C++慢慢分析,后面会慢慢更新。另外,博客只作为自己学习记录的一种方式,对于其他的不过多的评论。MediaPlayerDemopublic class MainActivity extends AppCompatActivity implements SurfaceHolder.Cal..._android多媒体播放源码分析 时序图

java 数据结构与算法 ——快速排序法-程序员宅基地

文章浏览阅读2.4k次,点赞41次,收藏13次。java 数据结构与算法 ——快速排序法_快速排序法