gorm中的關聯操作詳解

2022-10-09 21:01:09

一對一

belong to

屬於:可以理解為舔狗認為自己屬於女神,而女神都不知道舔狗的存在
type Girl struct { Id int Name string } type Dog struct { Id int Name string GirlId int Girl Girl } 遷移 schema db.AutoMigrate(&Dog{}) //此時會將Girl和Dog都生成

hasone

擁有:可以理解為女神知道舔狗的存在,舔狗只需要留一根「狗鏈子」在女神建立的時候交給女神就好了
type Dog struct { Id int Name string GirlId int } type Girl struct { Id int Name string Dog Dog } //db.AutoMigrate(&Girl{}) //只有一個Girl表生成 db.AutoMigrate(&Girl{}, &Dog{}) d := Dog{Id: 1, Name: "haha0"} db.Create(&Girl{Id: 1, Name: "nv0", Dog: d}) //此時Dog中的GirlId欄位被自動填寫

查詢

var girl Girl db.First(&girl, 1) fmt.Println(girl)
直接這樣查是不行的,這是需要先預載入
改成:db.Preload("Dog").First(&girl, 1)

關係操作

//給舔狗建立關係
db.Model(&Girl{Id:1}).Association("Dog").Append(&dog)
上面是hasone的情況,belong to 的情況正好相反
Append換為Delete,是刪除連線
換為Replace(&g1,&g2),是替換連線
Clear(),自身去掉所有關係

一對多

還是上面的例子只需要微微改動
` type Dog struct {
Id int
Name string
GirlId int
//Girl Girl
}

type Girl struct {
	Id   int
	Name string
	Dogs []Dog
}
d0 := Dog{Id: 1,
	Name: "haha0"}
d1 := Dog{Id: 2,
	Name: "haha2"}
g := &Girl{Id: 1,
	Name: "nv0",
	Dogs: []Dog{d0, d1},
}
db.Create(&g)
db.AutoMigrate(&Girl{}, &Dog{})

`
查詢:db.Preload("Dogs").First(&girl)
還是預載入,需要如果條件判斷

  • 類似內聯
    `
    // 帶條件的預載入 Order
    db.Preload("Orders", "state NOT IN (?)", "cancelled").Find(&users)
    // SELECT * FROM users;
    // SELECT * FROM orders WHERE user_id IN (1,2,3,4) AND state NOT IN ('cancelled');

db.Where("state = ?", "active").Preload("Orders", "state NOT IN (?)", "cancelled").Find(&users)
// SELECT * FROM users WHERE state = 'active';
// SELECT * FROM orders WHERE user_id IN (1,2) AND state NOT IN ('cancelled');

`

  • 自定義預載入
    db.Preload("Orders", func(db *gorm.DB) *gorm.DB { return db.Order("orders.amount DESC") }).Find(&users) // SELECT * FROM users; // SELECT * FROM orders WHERE user_id IN (1,2,3,4) order by orders.amount DESC;
  • 鏈式預載入
    修改結構體
    type Info struct { Id int money int GogId int } type Dog struct { Id int Name string GirlId int //Girl Girl Info Info }
    注意生成資料庫表的時候,各個表生成的順序,否則報錯
    db.AutoMigrate(&Girl{}, &Dog{}, &Info{})
    在查詢
    db.Preload("Dogs.Info").Preload("Dogs").First(&girl)
    同時加條件的話和上面也一樣,但是需要注意查詢條件只適用於她所在的預載入的那一層
    例如:db.Preload("Dogs.Info", "money>100").Preload("Dogs").First(&girl)
    輸出結果小於100的info資訊會沒有,但是dog依然會查出來,money>100條件影響不到Preload("Dogs")
    那怎麼樣可以實現輸出結果小於100的info資訊會沒有,dog也沒有呢,可以使用join

join

注意 Join適用於一對一的關係,例如: has one, belongs to
上面的問題可以:
db.Preload("Dogs", func(db *gorm.DB) *gorm.DB {
return db.Joins("Info").Where("money>200")
}).First(&girl)

多對多

建立格式:
type Info struct { Id int Money int DogId int } type Girl struct { Id int Name string Dogs []Doggorm:"many2many:girl_dog"} type Dog struct { Id int Name string Girl []Girlgorm:"many2many:girl_dog" Info Info }

查詢
1.只想查舔狗的女神有哪些,不要舔狗資訊

2.舔狗資訊也要,就加上預載入

想加條件和一對多差不多

關係維護
還是那幾個函數

刪除最好用事務

最後

注意多對多情況的FK和refer和一對一/多正好相反了

部落格是觀看b站up主(go圈裡最會寫js的奇淼)筆記
連結:https://www.bilibili.com/video/BV1E64y1472a?p=10&spm_id_from=pageDriver&vd_source=7de631e1910751f66cb6a72edf044dab