gorm 多对多关系

多对多关系,需要用第三张表存储两张表的关系

连接数据库

main.go

package main

import "gorm-demo/model"

func main() {
    model.InitMysql()
}

model/dbInit.go

package model

import (
    "fmt"
    "log"
    "os"
    "time"

    "gorm.io/driver/mysql"
    "gorm.io/gorm"
    "gorm.io/gorm/logger"
    "gorm.io/gorm/schema"
)

var Db *gorm.DB

const (
    Username    = "root"
    Password    = "123456"
    Host        = "127.0.0.1"
    Port        = "3306"
    DbName      = "gorm_demo" // 库名
    Config      = "charset=utf8mb4&parseTime=True&loc=Local"
    TablePrefix = "" // 表前缀t_
    // 数据库连接池最大空闲数
    MaxIdleConns = 10
    // 数据库连接池最大打开数
    MaxOpenConns = 100
    // 数据库连接池最大存活时间
    ConnMaxLifetime = time.Hour * 1
)

// 初始化数据库并产生数据库全局变量
func InitMysql() {
    //  默认的级别,会打印find找不到模型时的sql语句。
    // Silent 就不会。
    newLogger := logger.New(
        log.New(os.Stdout, "\r\n", log.LstdFlags), // io writer
        logger.Config{
            SlowThreshold:             time.Second, // 慢 SQL 阈值
            LogLevel:                  logger.Info, // Log level 可选 Silent,Error,Warn,Info
            Colorful:                  true,        // 禁用彩色打印 true:不禁用 false:禁用
            IgnoreRecordNotFoundError: true,        // 忽略ErrRecordNotFound(记录未找到)错误
        },
    )
    //"root:123456@tcp(127.0.0.1:3306)/gorm_demo?charset=utf8mb4&parseTime=True&loc=Local",
    dns := fmt.Sprintf("%s:%s@tcp(%s:%s)/%s?%s", Username, Password, Host, Port, DbName, Config)
    db, err := gorm.Open(mysql.New(mysql.Config{
        //注意:想要正确的处理 time.Time ,您需要带上 parseTime 参数, (更多参数) 要支持完整的 UTF-8 编码,您需要将 charset=utf8 更改为 charset=utf8mb4 查看 此文章 获取详情
        DSN:                       dns,   // DSN data source name
        DefaultStringSize:         256,   // string 类型字段的默认长度 utf8mb4长应该设置为171,否则如果以字符串作为主键,会报索引超长
        DisableDatetimePrecision:  true,  // 禁用 datetime 精度,MySQL 5.6 之前的数据库不支持
        DontSupportRenameIndex:    true,  // 重命名索引时采用删除并新建的方式,MySQL 5.7 之前的数据库和 MariaDB 不支持重命名索引
        DontSupportRenameColumn:   true,  // 用 `change` 重命名列,MySQL 8 之前的数据库和 MariaDB 不支持重命名列
        SkipInitializeWithVersion: false, // 根据当前 MySQL 版本自动配置                                                                                      //根据版本自动配置                                                                            // 根据当前 MySQL 版本自动配置
    }), &gorm.Config{
        //CreateBatchSize: 1000,    // 批量插入的条数
        SkipDefaultTransaction:                   false, //为了确保数据一致性,GORM 会在事务里执行写入操作(创建、更新、删除)。如果没有这方面的要求,您可以在初始化时禁用它。 true:禁用 / false 不禁用
        DisableForeignKeyConstraintWhenMigrating: true,  // 外键约束 在 AutoMigrate 或 CreateTable 时,GORM 会自动创建外键约束,若要禁用该特性,可将其设置为 true
        Logger:                                   newLogger,
        NamingStrategy: schema.NamingStrategy{
            TablePrefix:   TablePrefix, // 表名前缀,`User` 的表名应该是 `t_users`
            SingularTable: true,        // 使用单数表名,启用该选项,此时,`User` 的表名应该是 `t_user`
            NoLowerCase:   false,       // 禁用自动将字段名转换为小写
        },
    })
    if err != nil {
        fmt.Println("gorm.Open err:", err)
        panic("gorm.Open err")
    } else {
        Db = db
        sqlDB, _ := Db.DB()
        // Enable Logger, show detailed log
        sqlDB.SetMaxIdleConns(MaxIdleConns)       //设置空闲连接池中连接的最大数量
        sqlDB.SetMaxOpenConns(MaxOpenConns)       //设置打开数据库连接的最大数量。
        sqlDB.SetConnMaxLifetime(ConnMaxLifetime) //设置了连接可复用的最大时间。
        fmt.Println("")
        fmt.Println("InitMysql success")
        fmt.Println("")

        autoMigrate() //迁移文件
    }
}

// 迁移文件
func autoMigrate() {

    //belongstoDemo01()
    //belongstoDemo02()
    //belongstoDemo03()
    //belongstoDemo04()

    //hasOneDemo01()
    //hasOneDemo02()
    //hasOneDemo03()
    //hasOneDemo04()
    //hasOneDemo05()

    //hasmanyDemo01()
    //hasmanyDemo02()

    manyToManyDome01()
    //manyToManyDome02()
    //manyToManyDome03()
    //    实体关联

    //association_demo01()

    //testDemo01()
}

model/manyToMany.go

package model

import (
    "fmt"
    "gorm.io/gorm"
)

type Info struct {
    gorm.Model
    Money int
    DogID uint
}
type Dog struct {
    gorm.Model
    Name     string
    Info     Info
    GirlGods []GirlGod `gorm:"many2many:dog_girl_god"` //定义中间表dog_girl_god
}
type GirlGod struct {
    gorm.Model
    Name string
    Dogs []Dog `gorm:"many2many:dog_girl_god"` //定义中间表dog_girl_god
}

func manyToManyDome01() {
    Db.AutoMigrate(&Info{}, &Dog{}, &GirlGod{})

    //create01()
    //create02()

    //getGodsDome01()
    //查找关联
    //getGodsDome02()

    //修改关联关系 update Replace Delete
    updateDome01()

    //Delete with Select
    //删除记录的同时,也删除他所关联的数据
    //selectDelete()
}

//  创建狗狗1号,创建女神1,2号,并做关系
//  狗狗1号,女神1号和女神2号都会被创建,并在关联表里创建关联关系
func create01() {
    Info01 := Info{Money: 100}
    girlGod01 := GirlGod{Name: "GirlGod01"}
    girlGod02 := GirlGod{Name: "GirlGod02"}
    //write
    d := Dog{
        Name:     "dog01",
        GirlGods: []GirlGod{girlGod01, girlGod02},
        Info:     Info01,
    }

    Db.Create(&d)
}

// 创建狗狗2号,并关联女神1号和女神2号,

func create02() {
    // 方式一,前端传入女神1号和女神2号的id
    // 如果女神1号和女神2号已经存在,就不在创建了.
    //girlGod1 := GirlGod{
    //    Model: gorm.Model{ID: 1},
    //    Name:  "女神1号",
    //}
    //girlGod2 := GirlGod{
    //    Model: gorm.Model{ID: 2},
    //    Name:  "女神2号",
    //}
    //d02 := Dog{
    //    Name: "dog02",
    //    Info: Info{Money: 20000},
    //    GirlGods: []GirlGod{girlGod1, girlGod2},
    //}
    //Db.Create(&d02)

    //方式二 从表里查出要关联的女神
    var girlGods []GirlGod
    Db.Find(&girlGods, "id in (?)", []int{1, 2})
    d02 := Dog{
        Name:     "dog02",
        Info:     Info{Money: 20000}, //创建info表信息
        GirlGods: girlGods,
    }
    Db.Create(&d02)
}

func getGodsDome01() {
    var dog Dog
    Db.Find(&dog)
    fmt.Println(dog)
}
func getGodsDome02() {
    //查dogs表 预加载GirlGods
    //var dogs []Dog
    //Db.Preload("GirlGods").Find(&dogs)
    //fmt.Println(dogs)

    //查dogs表 预加载GirlGods和dogs的Info
    //var dogs []Dog
    //Db.Preload("Info").Preload("GirlGods").Find(&dogs)
    //fmt.Println(dogs)

    //查GirlGods 预加载Dogs和Dogs.Info
    //var girl []GirlGod
    //Db.Preload("Dogs.Info").Preload("Dogs").Find(&girl)
    //fmt.Println(girl)

    //查Dog表id为1的GirlGods信息(要的结果是girlGod信息,和GirlGods下的Dogs和Dogs.Info)
    //`dog` 是源模型,它的主键不能为空
    var girlGod []GirlGod
    //查Dog表,关联GirlGods表(Association是做关联查询)
    //Db.Model(&Dog{Model:gorm.Model{ID: 1}}).Association("GirlGods").Find(&girlGod)
    //Db.Model(&Dog{Model:gorm.Model{ID: 1}}).Preload("Dogs").Association("GirlGods").Find(&girlGod)
    //Db.Model(&Dog{Model:gorm.Model{ID: 1}}).Preload("Dogs.Info").Association("GirlGods").Find(&girlGod)
    //Db.Model(&Dog{Model:gorm.Model{ID: 1}}).Preload("Dogs.Info").Preload("Dogs").Association("GirlGods").Find(&girlGod)

    // 注意:Preload关联的表查出来的列表数据不能分页,想要分页,只能查关联表,关联表左右join,然后查出来再分页


    //查找带条件的关联
    //Db.Model(&dog).Where("girl_gods.name = GirlGod01").Association("GirlGods").Find(&girlGod)
    Db.Model(&Dog{Model: gorm.Model{ID: 1}}).
        //两行下面的where和Order是相对于girl_god表的操作
        Where("name in ?", []string{"GirlGod01", "GirlGod02"}).
        Order("id desc").
        Preload("Dogs.Info", func(db *gorm.DB) *gorm.DB {
            return db.Where("money > 100") //Info表
        }).
        Preload("Dogs", func(db *gorm.DB) *gorm.DB {
            return db.Order("id DESC") //dogs列表倒序
        }).
        Association("GirlGods").Find(&girlGod)
    fmt.Println(girlGod)

}

// 修改关联关系
func updateDome01() {
    g1 := GirlGod{Model: gorm.Model{ID: 5}}
    g2 := GirlGod{Model: gorm.Model{ID: 3}}
    dog := Dog{Model: gorm.Model{ID: 3}}

    //添加g1和dog3 还有g2和dog3的关系
    //Db.Model(&dog).Association("GirlGods").Append(&g1, &g2)
    //Db.Model(&dog).Association("GirlGods").Append(&g1)
    //Db.Model(&dog).Association("GirlGods").Append(&g2)

    //替换关联 会删除所有的关系,然后再插入关系
    Db.Model(&dog).Association("GirlGods").Replace(&g1, &g2)

    var gs = make([]GirlGod, 0) //空的切片,删除所有关系,不再写入任何关系
    dog1 := Dog{Model: gorm.Model{ID: 3}}
    Db.Model(&dog1).Association("GirlGods").Replace(&gs)

    // 关联计数 Count all GirlGods
    //count:=Db.Model(&dog).Association("GirlGods").Count()
    //fmt.Println(count)

    //  条件计数 Count dog.name in ("GirlGod01", "GirlGod02") all GirlGods
    //count:=Db.Model(&dog).Where("name IN ?", []string{"GirlGod01", "GirlGod02"}).Association("GirlGods").Count()
    //fmt.Println(count)

    //删除g1和dog3的关系
    //Db.Model(&dog).Association("GirlGods").Delete(&g1)

    //删除dog3的所有关系
    //Db.Model(&dog).Association("GirlGods").Clear()
}

//  Delete with Select
//  删除记录的同时,也删除他在关联表里的所有的关联关系
//  你也可以在删除记录时通过 Select 来删除具有 has one、has many、many2many 关系的记录,例如:
//  删除god3,同时也删除god3在关联表里的所有的关联关系
func selectDelete() {
    //dog3 :=Dog{Model:gorm.Model{ID: 3}}

    //Db.Select("GirlGods").Delete(&dog3) //删除dog3的同时,删除和GirlGods的所有关系
    //Db.Select("GirlGods","Info").Delete(&dog3) //删除dog3的同时,删除和GirlGods / Info的所有关系

    // 删除 dog3 时,也删除用户所有 has one/many、many2many 记录
    //Db.Select(clause.Associations).Delete(&dog3)

}

自定义连接表

默认的连接表,只有双方的主键id,展示不了更多信息了
自定义连接表 连接表 可以是一个功能齐全的模型,比如支持 软删除、钩子函数功能,并且可以具有更多字段。您可以通过 SetupJoinTable 设置

package model

import (
    "fmt"
    "gorm.io/gorm"
    "time"
)

//自定义连接表
//默认的连接表,只有双方的主键id,展示不了更多信息了
//自定义连接表 连接表 可以是一个功能齐全的模型,比如支持 软删除、钩子函数功能,并且可以具有更多字段。您可以通过 SetupJoinTable 设置

type Article struct {
    ID    uint
    Title string
    Tags  []Tag `gorm:"many2many:article_tags"`
}

type Tag struct {
    ID   uint
    Name string
}

type ArticleTag struct {
    ArticleID uint `gorm:"primaryKey"`
    TagID     uint `gorm:"primaryKey"`
    CreatedAt time.Time
}

func manyToManyDome03() {
    // 设置Article的Tags表为ArticleTag
    err := Db.SetupJoinTable(&Article{}, "Tags", &ArticleTag{})
    if err != nil {
        panic(err)
    }
    // 如果tag要反向应用Article,那么也得加上
    err = Db.SetupJoinTable(&Tag{}, "Articles", &ArticleTag{})
    if err != nil {
        panic(err)
    }
    Db.AutoMigrate(&Article{}, &Tag{}, &ArticleTag{})

    create3()
    //replace3()
    //getInfo()
}

// BeforeCreate 必须定义好所需的外键,否则会报错
func (p *ArticleTag) BeforeCreate(tx *gorm.DB) (err error) {
    //p.CreatedAt = time.Now() // CreatedAt time.Time 由于我们设置的是CreatedAt,gorm会自动填充当前时间,
    //p.Sort = 50  // 如果是其他的字段,需要使用到ArticleTag 的添加钩子 BeforeCreate 手动添加
    return nil
}

// 创建数据Title和tags 并添加关联数据
func create3() {
    Db.Create(&Article{
        Title: "flask零基础入门",
        Tags: []Tag{
            {Name: "python"},
            {Name: "后端"},
            {Name: "web"},
        },
    })

}

// 添加文章,关联已有标签
func create3_1() {
    Db.SetupJoinTable(&Article{}, "Tags", &ArticleTag{})
    var tags []Tag
    Db.Find(&tags, "name in ?", []string{"python", "web"})
    Db.Create(&Article{
        Title: "flask请求对象",
        Tags:  tags,
    })
}

// 给已有文章关联标签
func create3_2() {
    Db.SetupJoinTable(&Article{}, "Tags", &ArticleTag{})
    article := Article{
        Title: "django基础",
    }
    Db.Create(&article)
    var at Article
    var tags []Tag
    Db.Find(&tags, "name in ?", []string{"python", "web"})
    Db.Take(&at, article.ID).Association("Tags").Append(tags)
}

// 替换已有文章的标签 删除全部关联标签,写入新的标签
func replace3() {
    var article Article
    var tags []Tag
    //Db.Find(&tags, "name in ?", []string{"python1"}) // 查询出python1 如果查不到为空时,会删除全部关系。
    Db.Find(&tags, "name in ?", []string{"python"}) // 查询出python1 如果查不到为空时,会删除全部关系。
    Db.Take(&article, "title = ?", "flask零基础入门")
    Db.Model(&article).Association("Tags").Replace(tags)
}

// 查询文章列表,显示标签
/*
    SetupJoinTable
    添加和更新的时候得用这个
    这样才能走自定义的连接表,以及走它的钩子函数
    查询则不需要这个
*/
func getArticleWithTags() {
    var articles []Article
    Db.Preload("Tags").Find(&articles)
    fmt.Println(articles)
}

示列2

// setupJoinTable 自定义连接表
func setupJoinTable() {
    // 自定义连接表
    //设置Manufacturer的Platforms表为ManufacturerPlatforms
    err := DB.SetupJoinTable(&FaAdmin{}, "FaAuthGroups", &FaAuthGroupAccess{})
    if err != nil {
        logx.Error("db.SetupJoinTable FaAdmin FaAuthGroups table failed", zap.Error(err))
        panic("db.SetupJoinTable FaAdmin FaAuthGroups table failed")

    }
    // 设置Platform的Manufacturers表为ManufacturerPlatforms
    err = DB.SetupJoinTable(&FaAuthGroup{}, "FaAdmins", &FaAuthGroupAccess{})
    if err != nil {
        logx.Error("db.SetupJoinTable FaAuthGroup FaAdmins table failed", zap.Error(err))
        panic("db.SetupJoinTable FaAuthGroup FaAdmins table failed")
    }
}

重写中间表的外建

// FaAuthGroup 分组表
type FaAuthGroup struct {
    ID int64 `json:"id" gorm:"primaryKey;autoIncrement;comment:主键编码"`
    ControlTime
    Pid    int64  `json:"pid" gorm:"pid"`       // 父组别
    Name   string `json:"name" gorm:"name"`     // 组名
    Rules  string `json:"rules" gorm:"rules"`   // 规则ID
    Status string `json:"status" gorm:"status"` // 状态 hidden 禁用 normal:正常

    //多对多关系  fa_auth_group_access为关联表
    //joinForeignKey为 FaAuthGroup 表的主键id 关联 中间表fa_auth_group_access中的 group_id
    //joinReferences为 faAdmins表的主键 关联 中间表fa_auth_group_access 中的 uid
    FaAdmins []FaAdmin `json:"faAdmins" gorm:"comment:管理员;many2many:fa_auth_group_access;joinForeignKey:group_id;JoinReferences:uid"` // 仓库信息

}

数据表

中间表

package models

// FaAuthGroupAccess 权限分组表
type FaAuthGroupAccess struct {
    Uid          int64         `json:"uid" gorm:"uid"` // 会员ID
    FaAdmins     []FaAdmin     `json:"faAdmins" gorm:"foreignKey:ID;references:uid"`
    GroupId      int64         `json:"group_id" gorm:"group_id"` // 级别ID
    FaAuthGroups []FaAuthGroup `json:"faAuthGroups"   gorm:"foreignKey:ID;references:group_id"`
}

// TableName 表名称
func (*FaAuthGroupAccess) TableName() string {
    return "fa_auth_group_access"
}

func (t *FaAuthGroupAccess) Create(list []FaAuthGroupAccess) (err error) {
    err = DB.Create(&list).Error
    return err
}

CREATE TABLE `fa_auth_group_access` (
  `uid` int(10) unsigned NOT NULL COMMENT '会员ID',
  `group_id` int(10) unsigned NOT NULL COMMENT '级别ID',
  UNIQUE KEY `uid_group_id` (`uid`,`group_id`) USING BTREE,
  KEY `uid` (`uid`) USING BTREE,
  KEY `group_id` (`group_id`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC COMMENT='权限分组表';

分组表

package models

// FaAuthGroup 分组表
type FaAuthGroup struct {
    ID int64 `json:"id" gorm:"primaryKey;autoIncrement;comment:主键编码"`
    ControlTime
    Pid    int64  `json:"pid" gorm:"pid"`       // 父组别
    Name   string `json:"name" gorm:"name"`     // 组名
    Rules  string `json:"rules" gorm:"rules"`   // 规则ID
    Status string `json:"status" gorm:"status"` // 状态 hidden 禁用 normal:正常

    //多对多关系  fa_auth_group_access为关联表
    //joinForeignKey为 FaAuthGroup 表的主键id 关联 中间表fa_auth_group_access中的 group_id
    //joinReferences为 faAdmins表的主键 关联 中间表fa_auth_group_access 中的 uid
    FaAdmins []FaAdmin `json:"faAdmins" gorm:"comment:管理员;many2many:fa_auth_group_access;joinForeignKey:group_id;JoinReferences:uid"` // 仓库信息

}

// TableName 表名称
func (*FaAuthGroup) TableName() string {
    return "fa_auth_group"
}
CREATE TABLE `fa_auth_group` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `pid` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '父组别',
  `name` varchar(100) NOT NULL DEFAULT '' COMMENT '组名',
  `rules` text NOT NULL COMMENT '规则ID',
  `createtime` int(10) DEFAULT NULL COMMENT '创建时间',
  `updatetime` int(10) DEFAULT NULL COMMENT '更新时间',
  `status` varchar(30) NOT NULL DEFAULT '' COMMENT '状态',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=9 DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC COMMENT='分组表';

package models

import (
    "errors"
    "ytss_go_zero/app/core/cmd/rpc/pb"
    "ytss_go_zero/common/utils"
)

// FaAdmin 管理员表
type FaAdmin struct {
    ID int64 `json:"id" gorm:"primaryKey;autoIncrement;comment:主键编码"`
    ControlTime
    Username     string `json:"username" gorm:"username"`            // 用户名
    Nickname     string `json:"nickname" gorm:"nickname"`            // 昵称
    Password     string `json:"password" gorm:"password"`            // 密码
    Salt         string `json:"salt" gorm:"salt"`                    // 密码盐
    Department   string `json:"department" gorm:"column:department"` // 部门
    Avatar       string `json:"avatar" gorm:"avatar"`                // 头像
    Email        string `json:"email" gorm:"email"`                  // 电子邮箱
    Loginfailure int8   `json:"loginfailure" gorm:"loginfailure"`    // 失败次数
    Logintime    int64  `json:"logintime" gorm:"logintime"`          // 登录时间
    Loginip      string `json:"loginip" gorm:"loginip"`              // 登录IP
    Token        string `json:"token" gorm:"token"`                  // Session标识
    Status       string `json:"status" gorm:"status"`                // 状态 hidden 禁用 normal:正常

    //多对多关系  fa_auth_group_access为关联表
    //joinForeignKey为 FaAdmin表的主键id 关联 中间表fa_auth_group_access中的uid
    //joinReferences为 FaAuthGroup 关联 中间表fa_auth_group_access 中的 group_id
    FaAuthGroups []FaAuthGroup `json:"faAuthGroups" gorm:"comment:归属角色;many2many:fa_auth_group_access;joinForeignKey:uid;JoinReferences:group_id;"` // 仓库信息
}

// TableName 表名称
func (*FaAdmin) TableName() string {
    return "fa_admin"
}
CREATE TABLE `fa_admin` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID',
  `username` varchar(20) NOT NULL DEFAULT '' COMMENT '用户名',
  `nickname` varchar(50) NOT NULL DEFAULT '' COMMENT '昵称',
  `password` varchar(32) NOT NULL DEFAULT '' COMMENT '密码',
  `salt` varchar(30) NOT NULL DEFAULT '' COMMENT '密码盐',
  `department` varchar(50) DEFAULT NULL COMMENT '部门',
  `avatar` varchar(255) NOT NULL DEFAULT '' COMMENT '头像',
  `email` varchar(100) NOT NULL DEFAULT '' COMMENT '电子邮箱',
  `loginfailure` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '失败次数',
  `logintime` int(10) DEFAULT NULL COMMENT '登录时间',
  `loginip` varchar(50) DEFAULT NULL COMMENT '登录IP',
  `createtime` int(10) DEFAULT NULL COMMENT '创建时间',
  `updatetime` int(10) DEFAULT NULL COMMENT '更新时间',
  `token` varchar(59) NOT NULL DEFAULT '' COMMENT 'Session标识',
  `status` varchar(30) NOT NULL DEFAULT 'normal' COMMENT '状态',
  PRIMARY KEY (`id`) USING BTREE,
  UNIQUE KEY `username` (`username`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=37 DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC COMMENT='管理员表';
作者:海马  创建时间:2024-07-09 09:11
最后编辑:海马  更新时间:2025-01-27 10:55