goclub/sql
goclub/sql 让你了解每一个函数执行的sql是什么,保证SQL的性能最大化的同时超越ORM的便捷。
指南
在 Go 中与 sql 交互一般会使用 sqlx 或 gorm xorm
sqlx
偏底层,是对 database/sql
的封装,主要提供了基于结构体标签 db:"name""
将查询结果解析为结构体的功能。而GORM XORM 功能则更丰富。
直接使用 sqlx
频繁的手写 sql 非常繁琐且容易出错。(database/sql 的接口设计的不是很友好)。
GORM XORM 存在 ORM 都有的特点,使用者容易使用 ORM 运行一些性能不高的 SQL。虽然合理使用也可以写出高效SQL,但使用者在使用 ORM 的时候容易忽略最终运行的SQL是什么。
goclub/sql 提供介于手写 sql 和 ORM 之间的使用体验。
Open
连接数据库
goclub/sql 与 database/sql 连接方式相同,只是多返回了 dbClose 函数。 dbClose
等同于 db.Close
ExecMigrate
通过迁移代码创建表结构
定义Model
通过表单创建 Model: goclub.run
Insert
使用 Insert 插入数据
InsertModel
基于 Model 插入数据
大部分场景下使用 db.Insert
插入数据有点繁琐。基于 sq.Model
使用 db.InsertModel
操作数据会方便很多。
Update
使用 Update 更新数据
goclub/sql 故意没有提供 UpdateModel 方法, 因为使用 UpdateModel 性能并不好,会修改一下原本不需要修改的数据. 使用
db.Update(ctx, sq.QB{...})
可以”精准”的更新数据
Query
使用 Query 查询单条数据 使用 QuerySlice 查询多条数据
goclub/sql 特意没有提供 QueryModel 方法, 使用
db.Query(ctx, &user, sq.QB{ Where: sq.And(col.ID, sq.Equal(userID)) })
可以查询 Model
SoftDelete HardDelete
使用SoftDelete 或者 HardDelete 删除数据
Relation
Debug
sq.QB{
Debug: true
}
打开Debug可以查看
- 运行的SQL
- explain
- 执行时间
- last_query_cost
你也可以单独打开某一项或几项
sq.QB{
PrintSQL: true,
}
sq.QB{
Explain: true,
}
sq.QB{
RunTime: true,
}
sq.QB{
LastQueryCost: true,
}
Review
Review 的作用是用于审查 sql 或增加代码可读性
{#IN#}
语法: {#IN#}
默认会直接与执行SQL进行比对, 执行SQL与Review不一致则会在运行时 print 错误.
有时候执行的SQL不是固定的字符串例如
where in 时会根据查询条件不同导致有多种情况
select * from user where id in (?)
select * from user where id in (?,?)
select * from user where id in (?,?,?)
...
虽然可以使用 Reviews 配置多个review
sq.QB{
Review: []string{
"select * from user where id in (?)",
"select * from user where id in (?,?)",
"select * from user where id in (?,?,?),
},
}
但这样无法覆盖全部的情况.
可以使用 {#IN#}
模糊匹配
sq.QB{
Review: "select * from user where id in {#IN#}"
}
零次一次
语法
{{#任意字符#}}
如果你使用了 sq.IF
你可能需要用到 Reviews
sq.QB{
From: &User{},
Select: []sq.Column{"id"},
Where: sq.And("name", sq.IF(searchName != "", sq.Equal(searchName))),
Reviews: []string{
"SELECT `id` FROM `user` WHERE `name` = ? AND `deleted_at` IS NULL",
"SELECT `id` FROM `user` WHERE `deleted_at` IS NULL",
},
}
你可以使用 {{# and name = ?#}}
代替多个 review
建议将空格前置:使用 {{# and name = ?#}}
, 而不是 {{#and name = ? #}}
`
sq.QB{
From: &User{},
Select: []sq.Column{"id"},
Where: sq.And("name", sq.IF(searchName != "", sq.Equal(searchName))),
Review: "SELECT `id` FROM `user` WHERE{{# and name = ?#}} AND `deleted_at` IS NULL",
},
}
{#VALUES#}
语法:
{#VALUES#}
一些 Insert 语句会出现 (?,?)
(?,?),(?,?)
的情况
INSERT INTO `user` (`name`,`age`) VALUES (?,?),(?,?)
INSERT INTO `user` (`name`,`age`) VALUES (?,?)
可以使用 {#VALUES#}
模糊匹配
sq.QB{
Review: "INSERT INTO `user` (`name`,`age`) VALUES {#VALUES#}"
}
致谢
感谢 jetbrains 提供 Goland 开源授权