Object Relational Mapping, ORM 对象关系映射,用于将数据库中的数据与代码中的结构体进行映射,不同语言的实现方式略有区别,但是目的基本相同。
GoLang 中有多种映射库,这里简单介绍常见的 GORM 使用方式。
简介
ORM 实际上就是对数据库的操作进行封装,屏蔽数据库操作细节,从而简化开发,提高效率,GoLang 的 ORM 可以参考 gorm.io,其使用方法简单介绍如下。
GORM 倾向于使用约定而不是配置,默认使用 ID 作为主键,字段名的蛇型作为列名,结构体的蛇型复数作为表名,而且内部通过 gorm.Model
结构体提供了 CreatedAt
、UpdatedAt
字段跟踪创建、更新时间,通过 DeletedAt
作为逻辑删除时间。
例如结构体 UserInfo
实际转换为 user_infos
,字段名UserName
将转换为 user_name
,以此类推。
示例
如下是一个简单的示例,可以先手动创建表。
CREATE TABLE IF NOT EXISTS `users` (
`id` int NOT NULL AUTO_INCREMENT PRIMARY KEY,
`name` char(64) NOT NULL COMMENT '用户名',
`age` int NOT NULL COMMENT '用户的年龄',
`gender` enum('no','male','female') DEFAULT 'no' COMMENT '性别'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
INSERT INTO users(id, name, age, gender) VALUES(1, "andy", 18, "male");
CREATE TABLE IF NOT EXISTS `roles` (
`id` int NOT NULL AUTO_INCREMENT PRIMARY KEY,
`name` char(64) NOT NULL COMMENT '角色名'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
INSERT INTO roles(id, name) VALUES(1, "admin");
CREATE TABLE IF NOT EXISTS `map_user_role` (
`user_id` int NOT NULL,
`role_id` int NOT NULL,
PRIMARY KEY(user_id, role_id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
INSERT INTO map_user_role(user_id, role_id) VALUES(1, 1);
通过 GORM 执行 SQL 如下。
package main
import (
"log"
"gorm.io/driver/mysql"
"gorm.io/gorm"
dblog "gorm.io/gorm/logger"
)
type UserInfo struct {
ID int
Name string
Age int
Gender string
}
func (*UserInfo) TableName() string {
return "users"
}
func main() {
dsn := "root:msandbox@tcp(127.0.0.1:8027)/test?charset=utf8&parseTime=True&loc=Local"
db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{
Logger: dblog.Default.LogMode(dblog.Info),
})
if err != nil {
log.Printf("Open database failed, %v.", err)
return
}
//db.Debug().AutoMigrate(&UserInfo{}) // 迁移Schema字段类型未指定
// Create
db.Create(&UserInfo{Name: "Andy", Age: 18, Gender: "male"})
// 只写入指定字段,或者通过 Omit() 忽略字段
db.Select("Name", "Age").Create(&UserInfo{Name: "Andy", Age: 18, Gender: "male"})
// Read
var user UserInfo
db.First(&user, 1) // 根据主键查找
db.First(&user, "name = ?", "Andy") // 通过索引查找
// Update
db.Model(&user).Update("Price", 20) // 将 Age 更新为 20
db.Model(&user).Updates(&UserInfo{Name: "andy", Age: 20}) // 仅更新非零值字段
db.Model(&user).Updates(map[string]interface{}{"Name": "andy", "Age": "20"})
// Delete
db.Delete(&UserInfo{}, 1) // 根据ID删除
}
其中 gorm.io/driver/mysql
是 MySQL 驱动,实际上就是 github.com/go-sql-driver/mysql
只是进行了重新命名。
gorm 用 tag 来标识 MySQL 里面的约束,创建索引只需要直接指定列即可,如果需要多列组合索引,直接让索引的名字相同即可。
常用技巧
创建表
可以通过 db.HasTable()
来判断表是否存在,其入参可以使用两种形式:A) 字符串;B) 模型的地址类型。其判断方式是直接查询 INFORMATION_SCHEMA.TABLES
表中的数据。
if ok := DB.HasTable("foos"); ok {
t.Errorf("Table should not exist, but does")
}
if ok := DB.HasTable(&Foo{}); ok {
t.Errorf("Table should not exist, but does")
}
定义模型时,必须指定字段的首字母为大写,否则无法创建字段,同时可以使用 gorm tag
进行制定,可以参考 Declaring Models,不过有些调试起来比较复杂,还是直接创建比较好。
db.CreateTable(&User{})
r1 := db.DropTable("Users")
r2 := db.DropTable(&User{})
默认创建的表名为复数形式,例如 User
创建后的表名为 users
,如果不想创建复数形式的表名,可以通过如下的语句设置。
db.SingularTable(true)
如果要自己定义,可以通过如下方式修改。
type UserInfo struct {} // 默认表名是user_infos
// 设置UserInfo的表名为users
func (UserInfo) TableName() string {
return "users"
}
func (u UserInfo) TableName() string {
if u.Role == "admin" {
return "admin_users"
} else {
return "users"
}
}
更新时间
约定了 CreatedAt
UpdatedAt
DeletedAt
三个时间相关字段,会在创建、更新时自动填充当前时间。
----- 字段CreatedAt用于存储记录的创建时间
db.Create(&user) // 将会设置CreatedAt为当前时间
----- 要更改它的值, 需要使用Update
db.Model(&user).Update("CreatedAt", time.Now())
----- 字段UpdatedAt用于存储记录的修改时间
db.Save(&user) // 将会设置UpdatedAt为当前时间
db.Model(&user).Update("name", "jinzhu") // 将会设置UpdatedAt为当前时间
如果要使用不同的字段名,可以在 tag
中配置 autoCreateTime
、autoUpdateTime
标签,要保存时间戳而不是 time
,只需简单地将 time.Time
修改为 int
即可。
type User struct {
CreatedAt time.Time // Set to current time if it is zero on creating
UpdatedAt int // Set to current unix seconds on updating or if it is zero on creating
Updated int64 `gorm:"autoUpdateTime:nano"` // Use unix nano seconds as updating time
Updated int64 `gorm:"autoUpdateTime:milli"`// Use unix milli seconds as updating time
Created int64 `gorm:"autoCreateTime"` // Use unix seconds as creating time
}
注意,其中的 DeletedAt
字段用于保存记录删除时间,实际数据不会删除,只是更新该字段。
另外,像 MySQL 支持字段自动更新属性,那么就可以在 tag
中增加 gorm:"column:created_at;<-:false
设置不可写。
`created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
`updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
不过这样会有些兼容性问题。
时区
默认采用的是 UTC 时间,如果要修改为本地时区,那么可以在参数中增加 parseTime=True&loc=Local
即可,其中前者用于将数据库中的时间解析为 time.Time
类型,而后者是设置时区,还可以修改为 loc=Local
类型。
实现在 go-sql-driver/mysql/dsn.go
文件中,对应 Loc *time.Location
字段,默认是 time.UTC
时区。
嵌入结构体
上述示例中嵌入结构体可以直接使用,可以增加 gorm:"embedded"
显示表示嵌入,或者通过 gorm:"embedded;embeddedPrefix:author_"
指定前缀。
type Blog struct {
ID int
Author Author `gorm:"embedded;embeddedPrefix:author_"`
Upvotes int32
}
// 等价于
type Blog struct {
ID int64
AuthorName string
AuthorEmail string
Upvotes int32
}
连接池
使用的是标准库 database/sql
维护的连接池。
// 设置空闲连接池中连接的最大数量
db.SetMaxIdleConns(10)
// 设置打开数据库连接的最大数量
db.SetMaxOpenConns(100)
// 设置了连接可复用的最大时间
db.SetConnMaxLifetime(time.Hour)
高级技巧
Clause
GORM 内部会使用 SQL Builder 生成 SQL,而对于每个操作都会创建一个 *gorm.Statement
对象,所有的 GORM API 都是在为 Statement 添加或修改 Clause,最后根据这些 Clause 生成 SQL 语句。
例如,当执行 Limit
时会在 Statement 中添加以下 Clause 语句。
// Limit specify the number of records to be retrieved
func (db *DB) Limit(limit int) (tx *DB) {
tx = db.getInstance()
tx.Statement.AddClause(clause.Limit{Limit: limit})
return
}
然后在最后的 Callback 中构建最终的 SQL 语句,所以,可自定义 Clause 并与 GORM 一起使用。
不同数据库 Clause 可能会生成不同的 SQL 语句,例如:
db.Offset(10).Limit(5).Find(&users)
// SQL Server
SELECT * FROM "users" OFFSET 10 ROW FETCH NEXT 5 ROWS ONLY
// MySQL
SELECT * FROM `users` LIMIT 5 OFFSET 10
Upsert
MySQL 提供了 ON DUPLICATE KEY UPDATE
特有功能的支持,也就是如果 Insert 已经存在的记录,那么就执行 Update 操作,可以简化这一场景的实现逻辑,其它数据库也有类似的逻辑。
import "gorm.io/gorm/clause"
// 在冲突时,什么都不做
db.Clauses(clause.OnConflict{DoNothing: true}).Create(&user)
// 在`id`冲突时,将列更新为默认值
db.Clauses(clause.OnConflict{
Columns: []clause.Column{{Name: "id"}},
DoUpdates: clause.Assignments(map[string]interface{}{"role": "user"}),
}).Create(&users)
// INSERT INTO `users` *** ON DUPLICATE KEY UPDATE ***; MySQL
// 使用SQL语句
db.Clauses(clause.OnConflict{
Columns: []clause.Column{{Name: "id"}},
DoUpdates: clause.Assignments(map[string]interface{}{"count": gorm.Expr("GREATEST(count, VALUES(count))")}),
}).Create(&users)
// INSERT INTO `users` *** ON DUPLICATE KEY UPDATE `count`=GREATEST(count, VALUES(count));
// 在`id`冲突时,将列更新为新值
db.Clauses(clause.OnConflict{
Columns: []clause.Column{{Name: "id"}},
DoUpdates: clause.AssignmentColumns([]string{"name", "age"}),
}).Create(&users)
// INSERT INTO `users` *** ON DUPLICATE KEY UPDATE `name`=VALUES(name),`age=VALUES(age); MySQL
// 在冲突时,更新除主键以外的所有列到新值。
db.Clauses(clause.OnConflict{
UpdateAll: true,
}).Create(&users)
// INSERT INTO "users" *** ON CONFLICT ("id") DO UPDATE SET "age"="excluded"."age", ...;
详见 Gorm Upsert 中的介绍。