例如,您的应用包含了 user 和 language,且一个 user 可以说多种 language,多个 user 也可以说一种 language。

当使用 GORM 的 为 User 创建表时,GORM 会自动创建连接表

反向引用

  1. // User 拥有并属于多种 language,`user_languages` 是连接表
  2. type User struct {
  3. gorm.Model
  4. Languages []*Language `gorm:"many2many:user_languages;"`
  5. }
  6. type Language struct {
  7. gorm.Model
  8. Name string
  9. Users []*User `gorm:"many2many:user_languages;"`
  10. }

重写外键

对于 many2many 关系,连接表会同时拥有两个模型的外键,例如:

若要重写它们,可以使用标签 foreignKeyreferencesjoinforeignKeyjoinReferences。当然,您不需要使用全部的标签,你可以仅使用其中的一个重写部分的外键、引用。

  1. type User struct {
  2. gorm.Model
  3. Profiles []Profile `gorm:"many2many:user_profiles;foreignKey:Refer;joinForeignKey:UserReferID;References:UserRefer;joinReferences:ProfileRefer"`
  4. Refer uint `gorm:"index:,unique"`
  5. }
  6. type Profile struct {
  7. gorm.Model
  8. Name string
  9. UserRefer uint `gorm:"index:,unique"`
  10. // 会创建连接表:user_profiles
  11. // foreign key: user_refer_id, reference: users.refer

自引用 many2many 关系

预加载

GORM 可以通过 Preload 预加载 has many 关联的记录,查看 预加载 获取详情

Many2Many 的 CURD

查看 关联模式 获取 many2many 相关的用法

连接表 可以是一个全功能的模型,支持 Soft Delete钩子、更多的字段,就跟其它模型一样。您可以通过 SetupJoinTable 指定它,例如:

  1. type Person struct {
  2. ID int
  3. Name string
  4. Addresses []Address `gorm:"many2many:person_addresses;"`
  5. }
  6. type Address struct {
  7. ID uint
  8. Name string
  9. }
  10. type PersonAddress struct {
  11. PersonID int `gorm:"primaryKey"`
  12. AddressID int `gorm:"primaryKey"`
  13. CreatedAt time.Time
  14. DeletedAt gorm.DeletedAt
  15. }
  16. func (PersonAddress) BeforeCreate(db *gorm.DB) error {
  17. // ...
  18. }
  19. // 修改 Person 的 Addresses 字段的连接表为 PersonAddress
  20. // PersonAddress 必须定义好所需的外键,否则会报错
  21. err := db.SetupJoinTable(&Person{}, "Addresses", &PersonAddress{})

外键约束

你可以通过为标签 配置 OnUpdateOnDelete 实现外键约束,在使用 GORM 进行迁移时它会被创建,例如:

你也可以在删除记录时通过 Select 来删除 many2many 关系的记录,查看 Delete with Select 获取详情

复合外键

如果您的模型使用了 复合主键,GORM 会默认启用复合外键。

您也可以覆盖默认的外键、指定多个外键,只需用逗号分隔那些键名,例如:

  1. ID uint `gorm:"primaryKey"`
  2. Locale string `gorm:"primaryKey"`
  3. Value string
  4. }
  5. type Blog struct {
  6. ID uint `gorm:"primaryKey"`
  7. Locale string `gorm:"primaryKey"`
  8. Subject string
  9. Body string
  10. Tags []Tag `gorm:"many2many:blog_tags;"`
  11. LocaleTags []Tag `gorm:"many2many:locale_blog_tags;ForeignKey:id,locale;References:id"`
  12. SharedTags []Tag `gorm:"many2many:shared_blog_tags;ForeignKey:id;References:id"`
  13. }
  14. // 连接表:blog_tags
  15. // foreign key: blog_id, reference: blogs.id
  16. // foreign key: blog_locale, reference: blogs.locale
  17. // foreign key: tag_id, reference: tags.id
  18. // foreign key: tag_locale, reference: tags.locale
  19. // 连接表:locale_blog_tags
  20. // foreign key: blog_id, reference: blogs.id
  21. // foreign key: blog_locale, reference: blogs.locale
  22. // foreign key: tag_id, reference: tags.id
  23. // 连接表:shared_blog_tags
  24. // foreign key: tag_id, reference: tags.id