Bootstrap

GORM 一对一关系 Has one -Belongs to

一对一关系-Has One-belongs to

官方文档
has one与另一个模型建立一对一的关联,但它和一对一关系有些许不同。 这种关联表明一个模型的每个实例都包含或拥有另一个模型的一个实例。
例如User和IdCard,每个user只有一张IdCard

  一对一关系比较少,一般用于表的扩展
  例如一张用户表,有很多字段
  那么就可以把它拆分为两张表,常用的字段放主表,不常用的字段放详情表

表结构搭建-belongs to

我们建立两张表,一张用户表user,一张身份证表idcard
user表中有主键id和姓名,idcard中有对应的用户id以及其他

初始化相关表格语句:


package main

import (
	"fmt"
	"gorm.io/driver/mysql"
	"gorm.io/gorm"
	"gorm.io/gorm/logger"
)

var DB *gorm.DB

func init() {
	username := "root"
	password := "123456"
	host := "127.0.0.1"
	port := 3306
	Dbname := "gorm"
	timeout := "10s"

	dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8mb4&parseTime=True&loc=Local&timeout=%s", username, password, host, port, Dbname, timeout)
	db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{
		Logger: logger.Default.LogMode(logger.Info),
	})
	if err != nil {
		fmt.Println("连接数据库失败, error=", err)
		return
	}
	DB = db
	fmt.Println("数据库连接成功")
}

type User struct {
	ID   uint   `gorm:"size:16"`
	Name string `gorm:"size:8"`
}

type IDCard struct {
	ID     uint   `gorm:"size:16"`
	Age    int    `gorm:"size:4"`
	Addr   string `gorm:"size:16"`
	UserID uint   `gorm:"size:16"` //外键
	User   User   //关联的主表结构体
}

func main() {
	DB.AutoMigrate(&User{}, &IDCard{})
}

在这里插入图片描述

添加记录

//同时添加两个表的信息
//这里只能使用create,不能使用save(why?)	  
DB.Create(&User{
	ID:   1,
	Name: "wang2",
	IDCard: &IDCard{
		ID:   123,
		Age:  18,
		Addr: "shandong",
	},
})

//将新建的idcard绑定到已有user上
//DB.Save(&User{Name: "cc"})
DB.Save(&IDCard{ID: 456, UserID: 2})

查询

  为了方便从主表结构体查询子表内容,或者相互查询,可以采用类似如下的定义方法

type User struct {
	ID     uint    `gorm:"size:16"`
	Name   string  `gorm:"size:8"`
	IDCard *IDCard //通过idcard获得详细信息,进行后续查询,同时通过指针传递防止产生结构体嵌套
	//谁是指针不明确,怎么选择取决于具体的应用
}

type IDCard struct {
	ID     uint   `gorm:"size:16"`
	Age    int    `gorm:"size:4"`
	Addr   string `gorm:"size:16"`
	UserID uint   `gorm:"size:16"` //外键
	User   User   //关联的主表结构体
}

查询语句:

var user User
DB.Preload("IDCard").Take(&user, 2)
//{2 cc 0xc0001a8410}
marshal, _ := json.Marshal(user)//通过json转换显示指针对应内容
fmt.Println(string(marshal))
//{"ID":2,"Name":"cc","IDCard":{"ID":456,"Age":0,"Addr":"","UserID":2,"User":{"ID":0,"Name":"","IDCard":null}}}

删除

一对一的删除与多对多的相同,可以查看一对一的删除部分

;