模型定义

gorm.Model

gorm有默认的模型,包括IDCreatedAtUpdatedAtDeletedAt

  1. // 基本模型的定义
  2. type Model struct {
  3. ID uint `gorm:"primary_key"`
  4. CreatedAt time.Time
  5. UpdatedAt time.Time
  6. DeletedAt *time.Time
  7. }

可以将它嵌入到定义的模型中

  1. // 添加字段 `ID`, `CreatedAt`, `UpdatedAt`, `DeletedAt`
  2. type User struct {
  3. gorm.Model
  4. Name string
  5. }

表名

表名是结构体名称的复数形式

  1. type User struct {} // 默认表名是`users`
  2. // 设置User的表名为`profiles`
  3. func (User) TableName() string {
  4. return "profiles"
  5. }
  6. // 根据属性修改表名
  7. func (u User) TableName() string {
  8. if u.Role == "admin" {
  9. return "admin_users"
  10. } else {
  11. return "users"
  12. }
  13. }
  14. // 全局禁用表名复数
  15. db.SingularTable(true) // 如果设置为true,`User`的默认表名为`user`,使用`TableName`设置的表名不受影响

可以通过DefaultTableNameHandler对默认表名应用任何规则

  1. gorm.DefaultTableNameHandler = func (db *gorm.DB, defaultTableName string) string {
  2. return "prefix_" + defaultTableName;
  3. }

列名与字段

列名是字段名的蛇形小写

  1. type User struct {
  2. ID uint // 列名为 `id`
  3. Name string // 列名为 `name`
  4. Birthday time.Time // 列名为 `birthday`
  5. CreatedAt time.Time // 列名为 `created_at`
  6. }
  7. // 重设列名
  8. type Animal struct {
  9. AnimalId int64 `gorm:"column:beast_id"` // 设置列名为`beast_id`
  10. Birthday time.Time `gorm:"column:day_of_the_beast"` // 设置列名为`day_of_the_beast`
  11. Age int64 `gorm:"column:age_of_the_beast"` // 设置列名为`age_of_the_beast`
  12. }

字段ID为默认主键,可以通过 primary_key 来设置主键

  1. type User struct {
  2. ID uint // 字段`ID`为默认主键
  3. Name string
  4. }
  5. // 使用tag`primary_key`用来设置主键
  6. type Animal struct {
  7. AnimalId int64 `gorm:"primary_key"` // 设置AnimalId为主键
  8. Name string
  9. Age int64
  10. }

时间

  • CreatedAt用于存储创建时间
  • UpdatedAt用于存储记录的修改时间
  • DeletedAt用于存储记录的删除时间,如果记录存在 ``go db.Create(&user) // 将会设置CreatedAt`为当前时间

// 要更改它的值, 你需要使用Update db.Model(&user).Update(“CreatedAt”, time.Now())

db.Save(&user) // 将会设置UpdatedAt为当前时间 db.Model(&user).Update(“name”, “jinzhu”) // 将会设置UpdatedAt为当前时间

  1. > 删除具有`DeletedAt`字段的记录,它不会冲数据库中删除,但只将字段`DeletedAt`设置为当前时间,并在查询时无法找到记录,请参阅[软删除](http://gorm.book.jasperxu.com/crud.html#sd)
  2. <a name="TZm89"></a>
  3. ## 关联
  4. <a name="hhfuO"></a>
  5. ### 属于
  6. ```go
  7. // `User`属于`Profile`, `ProfileID`为外键(默认)
  8. type User struct {
  9. gorm.Model
  10. Profile Profile
  11. ProfileID int
  12. }
  13. type Profile struct {
  14. gorm.Model
  15. Name string
  16. }
  17. // SELECT * FROM profiles WHERE id = 111; // 111是user的外键ProfileID
  18. db.Model(&user).Related(&profile)

指定外键

  1. type Profile struct {
  2. gorm.Model
  3. Name string
  4. }
  5. type User struct {
  6. gorm.Model
  7. Profile Profile `gorm:"ForeignKey:ProfileRefer"` // 使用ProfileRefer作为外键
  8. ProfileRefer int
  9. }

指定外键和关联外键

  1. // Profile通过Refer找到关联的User
  2. type Profile struct {
  3. gorm.Model
  4. Refer string
  5. Name string
  6. }
  7. // User通过ProfileID找到关联的Profile
  8. type User struct {
  9. gorm.Model
  10. Profile Profile `gorm:"ForeignKey:ProfileID;AssociationForeignKey:Refer"`
  11. ProfileID int
  12. }

包含一个

  1. // User 包含一个 CreditCard, UserID 为外键
  2. type User struct {
  3. gorm.Model
  4. CreditCard CreditCard
  5. }
  6. type CreditCard struct {
  7. gorm.Model
  8. UserID uint
  9. Number string
  10. }
  11. var card CreditCard
  12. db.Model(&user).Related(&card, "CreditCard")
  13. //// SELECT * FROM credit_cards WHERE user_id = 123; // 123 is user's primary key
  14. // CreditCard是user的字段名称,这意味着获得user的CreditCard关系并将其填充到变量
  15. // 如果字段名与变量的类型名相同,如上例所示,可以省略,如:
  16. db.Model(&user).Related(&card)

指定外键

  1. type Profile struct {
  2. gorm.Model
  3. Name string
  4. UserRefer uint
  5. }
  6. type User struct {
  7. gorm.Model
  8. Profile Profile `gorm:"ForeignKey:UserRefer"`
  9. }

指定外键和关联外键

  1. type Profile struct {
  2. gorm.Model
  3. Name string
  4. UserID uint
  5. }
  6. type User struct {
  7. gorm.Model
  8. Refer string
  9. Profile Profile `gorm:"ForeignKey:UserID;AssociationForeignKey:Refer"`
  10. }

对比属于与包含一个,十分容易混淆

包含多个

  1. // User 包含多个 emails, UserID 为外键
  2. type User struct {
  3. gorm.Model
  4. Emails []Email
  5. }
  6. type Email struct {
  7. gorm.Model
  8. Email string
  9. UserID uint
  10. }
  11. db.Model(&user).Related(&emails)
  12. //// SELECT * FROM emails WHERE user_id = 111; // 111 是 user 的主键

指定外键

  1. type Profile struct {
  2. gorm.Model
  3. Name string
  4. UserRefer uint
  5. }
  6. type User struct {
  7. gorm.Model
  8. Profiles []Profile `gorm:"ForeignKey:UserRefer"`
  9. }

指定外键和关联外键

  1. type Profile struct {
  2. gorm.Model
  3. Name string
  4. UserID uint
  5. }
  6. type User struct {
  7. gorm.Model
  8. Refer string
  9. Profiles []Profile `gorm:"ForeignKey:UserID;AssociationForeignKey:Refer"`
  10. }

多对多

  1. // User 包含并属于多个 languages, 使用 `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. }
  10. db.Model(&user).Related(&languages, "Languages")
  11. //// SELECT * FROM "languages" INNER JOIN "user_languages" ON "user_languages"."language_id" = "languages"."id" WHERE "user_languages"."user_id" = 111

指定外键和关联外键

  1. type CustomizePerson struct {
  2. IdPerson string `gorm:"primary_key:true"`
  3. Accounts []CustomizeAccount `gorm:"many2many:PersonAccount;ForeignKey:IdPerson;AssociationForeignKey:IdAccount"`
  4. }
  5. type CustomizeAccount struct {
  6. IdAccount string `gorm:"primary_key:true"`
  7. Name string
  8. }

多种包含

  1. type Cat struct {
  2. Id int
  3. Name string
  4. Toy Toy `gorm:"polymorphic:Owner;"`
  5. }
  6. type Dog struct {
  7. Id int
  8. Name string
  9. Toy Toy `gorm:"polymorphic:Owner;"`
  10. }
  11. type Toy struct {
  12. Id int
  13. Name string
  14. OwnerId int
  15. OwnerType string
  16. }

多态属性和多对多显式不支持,并且会抛出错误。

关联模式

  1. // 开始关联模式
  2. var user User
  3. db.Model(&user).Association("Languages")
  4. // `user`是源,它需要是一个有效的记录(包含主键)
  5. // `Languages`是关系中源的字段名。
  6. // 如果这些条件不匹配,将返回一个错误,检查它:
  7. // db.Model(&user).Association("Languages").Error
  8. // Query - 查找所有相关关联
  9. db.Model(&user).Association("Languages").Find(&languages)
  10. // Append - 添加新的many2many, has_many关联, 会替换掉当前 has_one, belongs_to关联
  11. db.Model(&user).Association("Languages").Append([]Language{languageZH, languageEN})
  12. db.Model(&user).Association("Languages").Append(Language{Name: "DE"})
  13. // Delete - 删除源和传递的参数之间的关系,不会删除这些参数
  14. db.Model(&user).Association("Languages").Delete([]Language{languageZH, languageEN})
  15. db.Model(&user).Association("Languages").Delete(languageZH, languageEN)
  16. // Replace - 使用新的关联替换当前关联
  17. db.Model(&user).Association("Languages").Replace([]Language{languageZH, languageEN})
  18. db.Model(&user).Association("Languages").Replace(Language{Name: "DE"}, languageEN)
  19. // Count - 返回当前关联的计数
  20. db.Model(&user).Association("Languages").Count()
  21. // Clear - 删除源和当前关联之间的关系,不会删除这些关联
  22. db.Model(&user).Association("Languages").Clear()