快速入门
简介
Goravel 提供了一套非常简单易用的数据库交互方式,开发者可以使用 facades.Orm()
进行操作。目前,Goravel 为以下四种数据库提供了官方支持:
- MySQL 5.7+
- PostgreSQL 9.6+
- SQLite 3.8.8+
- SQL Server 2017+
在开始之前,请在 .env
文件中配置数据库链接信息,并确认 config/database.go
的默认配置。
配置
数据库的配置文件在 config/database.go
文件中。你可以在这个文件中配置所有的数据库连接,并指定默认的数据库连接。该文件中的大部分配置都基于项目的环境变量,且提供了 Goravel 所支持的数据库配置示例。
DSN
你也可以直接使用 DSN 连接数据库,只需要在配置文件中配置 dsn
字段即可:
"postgres": map[string]any{
"driver": "postgres",
++ "dsn": "postgres://user:password@localhost:5432/dbname?sslmode=disable",
...
}
读写分离
有时候您可能会希望使用一个数据库连接来执行 SELECT
语句,而 INSERT
、UPDATE
和 DELETE
语句则由另一个数据库连接来执行。在 Goravel 中可以轻松实现读写分离。
为了弄明白如何配置读写分离,我们先来看个例子:
import "github.com/goravel/framework/contracts/database"
// config/database.go
"connections": map[string]any{
"mysql": map[string]any{
"driver": "mysql",
"read": []database.Config{
{Host: "192.168.1.1", Port: 3306, Database: "forge", Username: "root", Password: "123123"},
},
"write": []database.Config{
{Host: "192.168.1.2", Port: 3306, Database: "forge", Username: "root", Password: "123123"},
},
"host": config.Env("DB_HOST", "127.0.0.1"),
"port": config.Env("DB_PORT", 3306),
"database": config.Env("DB_DATABASE", "forge"),
"username": config.Env("DB_USERNAME", ""),
"password": config.Env("DB_PASSWORD", ""),
"charset": "utf8mb4",
"loc": "Local",
},
}
我们在数据库配置中加入了两个键,分别是:read
, write
,192.168.1.1
将会被用作「读」连接主机,而 192.168.1.2
将作为「写」连接主机。这两个连接将共享 mysql
数组中的各项配置,如数据库前缀、字符编码等。如果 read
或 write
数组中存在多个值,Goravel 将会为每个连接随机选取所使用的数据库主机。
连接池
可以在配置文件中配置数据库连接池,合理的配置连接池参数,可以极大的提高并发性能:
配置键 | 作用 |
---|---|
pool.max_idle_conns | 最大空闲连接 |
pool.max_open_conns | 最大连接数 |
pool.conn_max_idletime | 连接最大空闲时间 |
pool.conn_max_lifetime | 连接最大生命周期 |
Schema
Postgres 和 Sqlserver 驱动支持配置 Schema。其中 Postgres 可以直接在配置文件中设置 Schema,而 Sqlserver 则需要通过在模型中设置 TableName
方法来指定 Schema。
Postgres
"connections": map[string]any{
"postgres": map[string]any{
"driver": "postgres",
...
"schema": "goravel",
},
}
Sqlserver
func (r *User) TableName() string {
return "goravel.users"
}
获取数据库信息
可以使用 db:show
命令查看数据库中的所有表。
go run . artisan db:show
也可以使用 db:table
命令查看指定表的结构。
go run . artisan db:table
go run . artisan db:table users
模型
模型相当于数据表的映射,你可以根据框架自带的模型文件 app/models/user.go
创建自定义模型。在 app/models/user.go
文件中 struct
嵌套了 orm.Model
与 orm.SoftDeletes
两个框架自带结构体,他们分别定义了 id, created_at, updated_at
与 deleted_at
,其中 orm.SoftDeletes
代表模型开启了软删除功能。
模型约定
- 模型使用大驼峰命名;
- 使用模型的复数形式「蛇形命名」来作为表名;
例如,模型名称为 UserOrder
,则表名为 user_orders
。
创建模型
go run . artisan make:model User
go run . artisan make:model user/User
指定表名
package models
import (
"github.com/goravel/framework/database/orm"
)
type User struct {
orm.Model
Name string
Avatar string
orm.SoftDeletes
}
func (r *User) TableName() string {
return "goravel_user"
}
数据库连接
默认情况下,所有模型使用的是应用程序配置的默认数据库连接。如果想指定在与特定模型交互时应该使用的不同连接,可以在模型上定义 Connection
方法:
package models
import (
"github.com/goravel/framework/database/orm"
)
type User struct {
orm.Model
Name string
Avatar string
orm.SoftDeletes
}
func (r *User) Connection() string {
return "postgres"
}
facades.Orm() 可用方法
方法名 | 作用 |
---|---|
Connection | 指定数据库链接 |
DB | 获取通用数据库接口 |
Query | 获取数据库实例 |
Transaction | 事务 |
WithContext | 注入 Context |
facades.Orm().Query() 可用方法
方法名 | 作用 |
---|---|
Begin | 手动开始事务 |
Commit | 提交事务 |
Count | 检索聚合 |
Create | 创建数据 |
Cursor | 游标 |
Delete | 删除数据 |
Distinct | 过滤重复 |
Driver | 获取当前驱动 |
Exec | 执行原生更新 SQL |
Exists | 数据是否存在 |
Find | 查询一条或多条数据 |
FindOrFail | 未找到时抛出错误 |
First | 查询一条数据 |
FirstOr | 查询或通过回调返回一条数据 |
FirstOrCreate | 查询或创建模型 |
FirstOrNew | 查询或实例化模型 |
FirstOrFail | 未找到时抛出错误 |
ForceDelete | 强制删除 |
Get | 查询多条数据 |
Group | Group 查询 |
Having | Having 查询 |
Join | Join 查询 |
Limit | 指定查询数量 |
LockForUpdate | 悲观锁 |
Model | 指定模型 |
Offset | 指定查询开始位置 |
Order | 排序 |
OrderBy | 排序 |
OrderByDesc | 排序 |
InRandomOrder | 排序 |
OrWhere | 查询条件 |
OrWhereNotIn | 查询条件 |
OrWhereNull | 查询条件 |
OrWhereIn | 查询条件 |
Paginate | 分页 |
Pluck | 查询单列 |
Raw | 执行原生查询 SQL |
Restore | 恢复软删除 |
Rollback | 手动回滚事务 |
Save | 保存修改 |
SaveQuietly | 静默的保存单个模型 |
Scan | 将数据解析到 struct |
Scopes | Scopes |
Select | 指定查询列 |
SharedLock | 悲观锁 |
Sum | 求和 |
Table | 指定表 |
ToSql | 获取 SQL |
ToRawSql | 获取 SQL |
Update | 更新单个字段 |
UpdateOrCreate | 更新或创建一条数据 |
Where | 查询条件 |
WhereBetween | 查询条件 |
WhereNotBetween | 查询条件 |
WhereNotIn | 查询条件 |
WhereNull | 查询条件 |
WhereIn | 查询条件 |
WithoutEvents | 静默事件 |
WithTrashed | 查询软删除 |
查询构造器
注入 Context
facades.Orm().WithContext(ctx)
指定数据库链接
如果你在配置文件 config/database.go
中定义了多个数据库连接,你可以通过 facades.Orm()
的 Connection
方法来使用它们。传递给 Connection
方法的连接名称应该是在 config/database.go
配置的连接之一:
facades.Orm().Connection("mysql")
获取通用数据库接口
获取通用数据库对象 sql.DB,然后使用其提供的功能:
db, err := facades.Orm().DB()
db, err := facades.Orm().Connection("mysql").DB()
// Ping
db.Ping()
// Close
db.Close()
// 返回数据库统计信息
db.Stats()
// SetMaxIdleConns 用于设置连接池中空闲连接的最大数量。
db.SetMaxIdleConns(10)
// SetMaxOpenConns 设置打开数据库连接的最大数量。
db.SetMaxOpenConns(100)
// SetConnMaxLifetime 设置了连接可复用的最大时间。
db.SetConnMaxLifetime(time.Hour)
获取数据库实例
每次进行具体数据库操作前,都需要先获取数据库的实例。
facades.Orm().Query()
facades.Orm().Connection("mysql").Query()
facades.Orm().WithContext(ctx).Query()
查询
查询一条数据
var user models.User
facades.Orm().Query().First(&user)
// SELECT * FROM `users` ORDER BY `users`.`id` LIMIT 1;
有时你可能希望检索查询的第一个结果或在未找到结果时执行一些其他操作。FirstOr
方法将返回匹配查询的第一个结果,或者,如果没有找到结果,则执行给定的闭包。你可以在闭包中对模型进行赋值:
facades.Orm().Query().Where("name", "first_user").FirstOr(&user, func() error {
user.Name = "goravel"
return nil
})
根据 ID 查询单条或多条数据
var user models.User
facades.Orm().Query().Find(&user, 1)
// SELECT * FROM `users` WHERE `users`.`id` = 1;
facades.Orm().Query().Find(&users, []int{1,2,3})
// SELECT * FROM `users` WHERE `users`.`id` IN (1,2,3);
未找到时抛出错误
var user models.User
err := facades.Orm().Query().FindOrFail(&user, 1)
当用户表主键为 string
类型,调用 Find
方法时需要指定主键
var user models.User
facades.Orm().Query().Find(&user, "uuid=?" ,"a")
// SELECT * FROM `users` WHERE `users`.`uuid` = "a";
查询多条数据
var users []models.User
facades.Orm().Query().Where("id in ?", []int{1,2,3}).Get(&users)
// SELECT * FROM `users` WHERE id in (1,2,3);
查询或创建模型
FirstOrCreate
方法将尝试使用给定的列 / 值对来查找数据库记录。如果在数据库中找不到该模型,则将插入一条记录,其中包含将第二个参数与可选的第三个参数合并后产生的属性:
FirstOrNew
方法,类似 FirstOrCreate
,会尝试在数据库中找到与给定属性匹配的记录。如果没有找到,则会返回一个新的模型实例。请注意,由 FirstOrNew
返回的模型尚未持久化到数据库中。需要手动调用 Save
方法来保存它:
var user models.User
facades.Orm().Query().Where("gender", 1).FirstOrCreate(&user, models.User{Name: "tom"})
// SELECT * FROM `users` WHERE `gender` = 1 AND `users`.`name` = 'tom' ORDER BY `users`.`id` LIMIT 1;
// INSERT INTO `users` (`created_at`,`updated_at`,`name`) VALUES ('2023-09-18 12:51:32.556','2023-09-18 12:51:32.556','tom');
facades.Orm().Query().Where("gender", 1).FirstOrCreate(&user, models.User{Name: "tom"}, models.User{Avatar: "avatar"})
// SELECT * FROM `users` WHERE `gender` = 1 AND `users`.`name` = 'tom' ORDER BY `users`.`id` LIMIT 1;
// INSERT INTO `users` (`created_at`,`updated_at`,`name`,`avatar`) VALUES ('2023-09-18 12:52:59.913','2023-09-18 12:52:59.913','tom','avatar');
var user models.User
facades.Orm().Query().Where("gender", 1).FirstOrNew(&user, models.User{Name: "tom"})
// SELECT * FROM `users` WHERE `gender` = 1 AND `users`.`name` = 'tom' ORDER BY `users`.`id` LIMIT 1;
facades.Orm().Query().Where("gender", 1).FirstOrNew(&user, models.User{Name: "tom"}, models.User{Avatar: "avatar"})
// SELECT * FROM `users` WHERE `gender` = 1 AND `users`.`name` = 'tom' ORDER BY `users`.`id` LIMIT 1;
未找到时抛出错误
当找不到模型时,First
方法不会抛出错误,如果想抛出,可以使用 FirstOrFail
:
var user models.User
err := facades.Orm().Query().FirstOrFail(&user)
// err == orm.ErrRecordNotFound
Where 条件
facades.Orm().Query().Where("name", "tom")
facades.Orm().Query().Where("name = 'tom'")
facades.Orm().Query().Where("name = ?", "tom")
facades.Orm().Query().WhereBetween("age", 1, 10)
facades.Orm().Query().WhereNotBetween("age", 1, 10)
facades.Orm().Query().WhereNotIn("name", []any{"a"})
facades.Orm().Query().WhereNull("name")
facades.Orm().Query().WhereIn("name", []any{"a"})
facades.Orm().Query().OrWhere("name = ?", "tom")
facades.Orm().Query().OrWhereNotIn("name", []any{"a"})
facades.Orm().Query().OrWhereNUll("name")
facades.Orm().Query().OrWhereIn("name", []any{"a"})
指定查询数量
var users []models.User
facades.Orm().Query().Where("name = ?", "tom").Limit(3).Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' LIMIT 3;
指定查询开始位置
var users []models.User
facades.Orm().Query().Where("name = ?", "tom").Offset(5).Limit(3).Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' LIMIT 3 OFFSET 5;
排序
var users []models.User
facades.Orm().Query().Where("name = ?", "tom").Order("sort asc").Order("id desc").Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' ORDER BY sort asc,id desc;
facades.Orm().Query().Where("name = ?", "tom").OrderBy("sort").Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' ORDER BY sort asc;
facades.Orm().Query().Where("name = ?", "tom").OrderBy("sort", "desc").Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' ORDER BY sort desc;
facades.Orm().Query().Where("name = ?", "tom").OrderByDesc("sort").Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' ORDER BY sort desc;
facades.Orm().Query().Where("name = ?", "tom").InRandomOrder().Get(&users)
// SELECT * FROM `users` WHERE name = 'tom' ORDER BY RAND();
分页
var users []models.User
var total int64
facades.Orm().Query().Paginate(1, 10, &users, &total)
// SELECT count(*) FROM `users`;
// SELECT * FROM `users` LIMIT 10;
查询单列
var ages []int64
facades.Orm().Query().Model(&models.User{}).Pluck("age", &ages)
// SELECT `age` FROM `users`;
指定表查询
如果想查询一些聚合数据,需要指定具体表。
使用模型指定
var count int64
facades.Orm().Query().Model(&models.User{}).Count(&count)
// SELECT count(*) FROM `users` WHERE deleted_at IS NULL;
使用表名指定
var count int64
facades.Orm().Query().Table("users").Count(&count)
// SELECT count(*) FROM `users`; // get all records, whether deleted or not
获取 SQL
带占位符的 SQL:
facades.Orm().Query().ToSql().Get(models.User{})
// SELECT * FROM "users" WHERE "id" = $1 AND "users"."deleted_at" IS NULL
带绑定值的 SQL:
facades.Orm().Query().ToRawSql().Get(models.User{})
// SELECT * FROM "users" WHERE "id" = 1 AND "users"."deleted_at" IS NULL
ToSql
与 ToRawSql
后可以调用的方法:Count
, Create
, Delete
, Find
, First
, Get
, Pluck
, Save
, Sum
, Update
。
检索聚合
var count int
facades.Orm().Query().Where("name = ?", "tom").Count(&count)
// SELECT count(*) FROM `users` WHERE name = 'tom';
指定查询列
Select
允许你指定从数据库中检索哪些字段,默认情况下,ORM 会检索所有字段。
facades.Orm().Query().Select("name", "age").Get(&users)
// SELECT `name`,`age` FROM `users`;
facades.Orm().Query().Select([]string{"name", "age"}).Get(&users)
// SELECT `name`,`age` FROM `users`;
Group By & Having
type Result struct {
Name string
Total int
}
var result Result
facades.Orm().Query().Model(&models.User{}).Select("name, sum(age) as total").Group("name").Having("name = ?", "tom").Get(&result)
// SELECT name, sum(age) as total FROM `users` GROUP BY `name` HAVING name = "tom";
Join 查询
type Result struct {
Name string
Email string
}
var result Result
facades.Orm().Query().Model(&models.User{}).Select("users.name, emails.email").Join("left join emails on emails.user_id = users.id").Scan(&result)
// SELECT users.name, emails.email FROM `users` LEFT JOIN emails ON emails.user_id = users.id;
创建
user := models.User{Name: "tom", Age: 18}
err := facades.Orm().Query().Create(&user)
// INSERT INTO users (name, age, created_at, updated_at) VALUES ("tom", 18, "2022-09-27 22:00:00", "2022-09-27 22:00:00");
// 不触发模型事件
err := facades.Orm().Query().Table("users").Create(map[string]any{
"name": "Goravel",
})
// 触发模型事件
err := facades.Orm().Query().Model(&models.User{}).Create(map[string]any{
"name": "Goravel",
})
批量创建
users := []models.User{{Name: "tom", Age: 18}, {Name: "tim", Age: 19}}
err := facades.Orm().Query().Create(&users)
err := facades.Orm().Query().Table("users").Create(&[]map[string]any{
{"name": "Goravel"},
{"name": "Framework"},
})
err := facades.Orm().Query().Model(&models.User{}).Create(&[]map[string]any{
{"name": "Goravel"},
{"name": "Framework"},
})
created_at
和updated_at
字段将会被自动填充。
游标
可用于在查询数万条模型记录时减少内存的使用。注意,Cursor
无法与预加载 With
一同使用,请在 for
循环中使用延迟预加载实现加载关联数据。
cursor, err := facades.Orm().Query().Model(models.User{}).Cursor()
if err != nil {
return err
}
for row := range cursor {
var user models.User
if err := row.Scan(&user); err != nil {
return err
}
fmt.Println(user)
}
更新
在现有模型基础上进行更新
var user models.User
facades.Orm().Query().First(&user)
user.Name = "tom"
user.Age = 100
facades.Orm().Query().Save(&user)
// UPDATE `users` SET `created_at`='2023-09-14 16:03:29.454',`updated_at`='2023-09-18 21:05:59.896',`name`='tom',`age`=100,`avatar`='' WHERE `id` = 1;
更新单一字段
facades.Orm().Query().Model(&models.User{}).Where("name", "tom").Update("name", "hello")
// UPDATE `users` SET `name`='hello',`updated_at`='2023-09-18 21:06:30.373' WHERE `name` = 'tom';
facades.Orm().Query().Model(&models.User{}).Where("name", "tom").Update(models.User{Name: "hello", Age: 18})
// UPDATE `users` SET `updated_at`='2023-09-18 21:07:06.489',`name`='hello',`age`=18 WHERE `name` = 'tom';
当使用
struct
进行批量更新时,Orm 只会更新非零值的字段。你可以使用map
更新字段,或者使用Select
指定要更新的字段。注意struct
只能为Model
,如果想用非Model
批量更新,需要使用.Table("users")
,但此时无法自动更新updated_at
字段。
更新或创建一条数据
根据 name
查询,如果不存在,则根据 name
, avatar
创建,如果存在,则根据 name
更新 avatar
:
facades.Orm().Query().UpdateOrCreate(&user, models.User{Name: "name"}, models.User{Avatar: "avatar"})
// SELECT * FROM `users` WHERE `users`.`name` = 'name' AND `users`.`deleted_at` IS NULL ORDER BY `users`.`id` LIMIT 1;
// INSERT INTO `users` (`created_at`,`updated_at`,`deleted_at`,`name`,`avatar`) VALUES ('2023-03-11 10:11:08.869','2023-03-11 10:11:08.869',NULL,'name','avatar');
// UPDATE `users` SET `name`='name',avatar`='avatar',`updated_at`='2023-03-11 10:11:08.881' WHERE users`.`deleted_at` IS NULL AND `id` = 1;
删除
根据模型删除,该方法将返回受影响的行数:
var user models.User
err := facades.Orm().Query().Find(&user, 1)
res, err := facades.Orm().Query().Delete(&user)
res, err := facades.Orm().Query().Model(&models.User{}).Where("id", 1).Delete()
res, err := facades.Orm().Query().Table("users").Where("id", 1).Delete()
// DELETE FROM `users` WHERE `users`.`id` = 1;
num := res.RowsAffected
批量删除
facades.Orm().Query().Where("name = ?", "tom").Delete(&models.User{})
// DELETE FROM `users` WHERE name = 'tom';
如果模型开启了软删除功能,想要强制删除某数据
facades.Orm().Query().Where("name", "tom").ForceDelete(&models.User{})
facades.Orm().Query().Model(&models.User{}).Where("name", "tom").ForceDelete()
facades.Orm().Query().Table("users").Where("name", "tom").ForceDelete()
您可以通过 Select
来删除具有模型关联的记录:
// 删除 user 时,也删除 user 的 account
facades.Orm().Query().Select("Account").Delete(&user)
// 删除 user 时,也删除 user 的 Orders、CreditCards 记录
facades.Orm().Query().Select("Orders", "CreditCards").Delete(&user)
// 删除 user 时,也删除用户所有子关联
facades.Orm().Query().Select(orm.Associations).Delete(&user)
// 删除 users 时,也删除每一个 user 的 account
facades.Orm().Query().Select("Account").Delete(&users)
注意:只有当记录的主键不为空时,关联才会被删除,Orm 会使用这些主键作为条件来删除关联记录:
// 会删除所有 name=`goravel` 的 user,但这些 user 的 account 不会被删除
facades.Orm().Query().Select("Account").Where("name = ?", "goravel").Delete(&models.User{})
// 会删除 name = `goravel` 且 id = `1` 的 user,并且 user `1` 的 account 也会被删除
facades.Orm().Query().Select("Account").Where("name = ?", "goravel").Delete(&models.User{ID: 1})
// 会删除 id = `1` 的 user,并且 account 也会被删除
facades.Orm().Query().Select("Account").Delete(&models.User{ID: 1})
如果在没有任何条件的情况下执行批量删除,ORM 不会执行该操作,并返回错误。对此,你必须加一些条件,或者使用原生 SQL。
查询软删除
var user models.User
facades.Orm().Query().WithTrashed().First(&user)
过滤重复
var users []models.User
facades.Orm().Query().Distinct("name").Find(&users)
获取当前驱动
driver := facades.Orm().Query().Driver()
// 判断驱动
if driver == orm.DriverMysql {}
执行原生查询 SQL
type Result struct {
ID int
Name string
Age int
}
var result Result
facades.Orm().Query().Raw("SELECT id, name, age FROM users WHERE name = ?", "tom").Scan(&result)
执行原生更新 SQL
该方法将返回受影响的行数:
res, err := facades.Orm().Query().Exec("DROP TABLE users")
// DROP TABLE `users`;
num := res.RowsAffected
数据是否存在
var exists bool
facades.Orm().Query().Model(&models.User{}).Where("name", "tom").Exists(&exists)
恢复软删除
facades.Orm().Query().WithTrashed().Restore(&models.User{ID: 1})
facades.Orm().Query().Model(&models.User{ID: 1}).WithTrashed().Restore()
// UPDATE `users` SET `deleted_at`=NULL WHERE `id` = 1;
事务
可以使用 Transaction
方法执行事务:
import (
"github.com/goravel/framework/contracts/database/orm"
"github.com/goravel/framework/facades"
"goravel/app/models"
)
...
return facades.Orm().Transaction(func(tx orm.Query) error {
var user models.User
return tx.Find(&user, user.ID)
})
也可以自己手动控制事务的流程:
tx, err := facades.Orm().Query().Begin()
user := models.User{Name: "Goravel"}
if err := tx.Create(&user); err != nil {
err := tx.Rollback()
} else {
err := tx.Commit()
}
Scopes
允许你指定常用的查询,可以在调用方法时引用这些查询。
func Paginator(page string, limit string) func(methods orm.Query) orm.Query {
return func(query orm.Query) orm.Query {
page, _ := strconv.Atoi(page)
limit, _ := strconv.Atoi(limit)
offset := (page - 1) * limit
return query.Offset(offset).Limit(limit)
}
}
// scopes.Paginator 是一个自定义的 func(ormcontract.Query) ormcontract.Query
facades.Orm().Query().Scopes(scopes.Paginator(page, limit)).Find(&entries)
原生表达式
可以使用 db.Raw
方法进行字段的更新:
import "github.com/goravel/framework/database/db"
facades.Orm().Query().Model(&user).Update("age", db.Raw("age - ?", 1))
// UPDATE `users` SET `age`=age - 1,`updated_at`='2023-09-14 14:03:20.899' WHERE `users`.`deleted_at` IS NULL AND `id` = 1;
悲观锁
查询构建器还包括一些函数,可帮助您在执行 select
语句时实现「悲观锁」。
您可以调用 SharedLock
方法使用「共享锁」执行语句,共享锁可防止选定的行被修改,直到您的事务被提交:
var users []models.User
facades.Orm().Query().Where("votes", ">", 100).SharedLock().Get(&users)
或者,您可以使用 LockForUpdate
方法。该锁可防止所选记录被修改或被另一个共享锁选中:
var users []models.User
facades.Orm().Query().Where("votes", ">", 100).LockForUpdate().Get(&users)
求和
var sum int
if err := facades.Orm().Query().Model(models.User{}).Sum("id", &sum); err != nil {
return err
}
fmt.Println(sum)
Events
Orm 模型触发几个事件,允许你挂接到模型生命周期的如下节点:Retrieved
、Creating
、Created
、Updating
、Updated
、Saving
、Saved
、Deleting
、Deleted
、ForceDeleting
、ForceDeleted
、Restored
、Restoring
。
当从数据库中检索到现有模型时,将调度 Retrieved
事件。当一个新模型第一次被保存时,Creating
和 Created
事件将被触发。 Updating
/ Updated
事件将在修改现有模型并调用 Save
方法时触发。Saving
/ Saved
事件将在创建或更新模型时触发 - 即使模型的属性没有更改。以「-ing」结尾的事件名称在模型的任何更改被持久化之前被调度,而以「-ed」结尾的事件在对模型的更改被持久化之后被调度。
要开始监听模型事件,请在模型上定义一个 DispatchesEvents
方法。此方法将模型生命周期的各个点映射到您定义的事件类中。
import (
contractsorm "github.com/goravel/framework/contracts/database/orm"
"github.com/goravel/framework/database/orm"
)
type User struct {
orm.Model
Name string
}
func (u *User) DispatchesEvents() map[contractsorm.EventType]func(contractsorm.Event) error {
return map[contractsorm.EventType]func(contractsorm.Event) error{
contractsorm.EventCreating: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventCreated: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventSaving: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventSaved: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventUpdating: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventUpdated: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventDeleting: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventDeleted: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventForceDeleting: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventForceDeleted: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventRetrieved: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventRestored: func(event contractsorm.Event) error {
return nil
},
contractsorm.EventRestoring: func(event contractsorm.Event) error {
return nil
},
}
}
注意:仅注册用到的事件即可。通过 Orm 进行批量操作时,不会调度模型事件。
观察者
定义观察者
如果在一个模型上监听了多个事件,可以使用观察者来将这些监听器组织到一个单独的类中。观察者类的方法名映射到你希望监听的事件。make:observer
Artisan 命令可以快速建立新的观察者类:
go run . artisan make:observer UserObserver
go run . artisan make:observer user/UserObserver
此命令将在 app/observers
文件夹放置新的观察者类。如果这个目录不存在,Artisan 将替您创建:
package observers
import (
"fmt"
"github.com/goravel/framework/contracts/database/orm"
)
type UserObserver struct{}
func (u *UserObserver) Created(event orm.Event) error {
return nil
}
func (u *UserObserver) Updated(event orm.Event) error {
return nil
}
func (u *UserObserver) Deleted(event orm.Event) error {
return nil
}
func (u *UserObserver) ForceDeleted(event orm.Event) error {
return nil
}
模版中仅包含部分事件,可以根据需要手动添加其他事件。
要注册观察者,需要将观察者与要观察的模型绑定。您可以在 app/providers/event_service_provider.go::Boot
方法中注册观察者:
package providers
import (
"github.com/goravel/framework/facades"
"goravel/app/models"
"goravel/app/observers"
)
type EventServiceProvider struct {
}
func (receiver *EventServiceProvider) Register(app foundation.Application) {
facades.Event().Register(receiver.listen())
}
func (receiver *EventServiceProvider) Boot(app foundation.Application) {
facades.Orm().Observe(models.User{}, &observers.UserObserver{})
}
func (receiver *EventServiceProvider) listen() map[event.Event][]event.Listener {
return map[event.Event][]event.Listener{}
}
注意:如果同时使用了
DispatchesEvents
与Observer
,将只应用DispatchesEvents
。
观察者传参
所有的事件默认传入 event
参数,包含以下方法:
方法名 | 作用 |
---|---|
Context | 获取 facades.Orm().WithContext() 传入的 context |
GetAttribute | 获取修改的值,如未修改,获取原始值,如没有原始值,返回 nil |
GetOriginal | 获取原始值,如没有原始值,返回 nil |
IsDirty | 判断字段是否修改 |
IsClean | IsDirty 取反 |
Query | 获取一个新的 Query,可以配合事务使用 |
SetAttribute | 为字段设置一个新值 |
静默事件
也许有时候你会需要暂时将所有由模型触发的事件「静默」处理,可以使用 WithoutEvents
方法:
var user models.User
facades.Orm().Query().WithoutEvents().Find(&user, 1)
静默的保存单个模型
有时候,你也许会想要「保存」一个已有的模型,且不触发任何事件。那么你可用 SaveQuietly
方法:
var user models.User
err := facades.Orm().Query().FindOrFail(&user, 1)
user.Name = "Goravel"
err := facades.Orm().Query().SaveQuietly(&user)