如何在 Gorm 中附加到多对多关系而不更新插入已经存在的关联行?

ech*_*315 9 orm go go-gorm

如何在多对多关联中将模型与已存在的其他模型关联起来,而不需要 GORM 执行查询来 UPSERT 已存在的模型?

例子:

假设我有两个相互多对多关联的 GORM 模型:

type A struct {
    ID int `gorm:"primaryKey"`
    Bs []B `gorm:"many2many:a_bs;"`
}

type B struct {
    ID int `gorm:"primaryKey"`
    As []A `gorm:"many2many:a_bs;"`
}
Run Code Online (Sandbox Code Playgroud)

假设我创建两个 As:

a1 := A{}
db.Create(&a1)

a2 := A{}
db.Create(&a2)
Run Code Online (Sandbox Code Playgroud)

这将按预期产生以下查询:

INSERT INTO "as" DEFAULT VALUES RETURNING "id"
INSERT INTO "as" DEFAULT VALUES RETURNING "id"
Run Code Online (Sandbox Code Playgroud)

假设我想创建一个 B 并将其与 As 关联:

b := B{
    As: []A{a1, a2},
}
db.Create(&b)
Run Code Online (Sandbox Code Playgroud)

这将导致以下查询:

INSERT INTO "as" ("id") VALUES (3),(4) ON CONFLICT DO NOTHING RETURNING "id"
INSERT INTO "a_bs" ("b_id","a_id") VALUES (3,3),(3,4) ON CONFLICT DO NOTHING
INSERT INTO "bs" DEFAULT VALUES RETURNING "id"
Run Code Online (Sandbox Code Playgroud)

如何消除第一个查询,该查询不必要地更新插入已插入的 A 模型?

使用

b := B{}
db.Create(&b)
db.Model(&b).Association("As").Append([]A{a1, a2})
Run Code Online (Sandbox Code Playgroud)

第二位代码具有相同的效果。添加 Omit("As") 子句将取消整个插入。

Eze*_*uns 4

db.Omit("As.*").Create(&b)
Run Code Online (Sandbox Code Playgroud)

gorm Associations 上有一个注释:Skip Auto Create/Update文档,其中详细介绍了这一点。很容易错过。