如何用go实现一个ORM https://mp.weixin.qq.com/s/06pZl4GpM0wAnyZmn7Hjfw
如何用go实现一个ORM
原创 洪胜杰 哔哩哔哩技术 2023-02-21 12:00 发表于上海 收录于合集 #工程提效1个 #开发工具1个 #后端研发5个本期作者
洪胜杰
B端技术中心高级开发工程师
为了提高开发效率和质量,我们常常需要ORM来帮助我们快速实现持久层增删改查API,目前go语言实现的ORM有很多种,他们都有自己的优劣点,有的实现简单,有的功能复杂,有的API十分优雅。在使用了多个类似的工具之后,总是会发现某些点无法满足解决我们生产环境中碰到的实际问题,比如无法集成公司内部的监控,Trace组件,没有database层的超时设置,没有熔断等,所以有必要公司自己内部实现一款满足我们可自定义开发的ORM,好用的生产工具常常能够对生产力产生飞跃式的提升。
为什么需要ORM
直接使用database/sql的痛点
首先看看用database/sql如何查询数据库
我们用user表来做例子,一般的工作流程是先做技术方案,其中排在比较前面的是数据库表的设计,大部分公司应该有严格的数据库权限控制,不会给线上程序使用比较危险的操作权限,比如创建删除数据库,表,删除数据等。
表结构如下:
CREATE TABLE `user` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT 'id',
`name` varchar(100) NOT NULL COMMENT '名称',
`age` int(11) NOT NULL DEFAULT '0' COMMENT '年龄',
`ctime` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`mtime` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '更新时间',
PRIMARY KEY (`id`),
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
首先我们要写出和表结构对应的结构体User,如果你足够勤奋和努力,相应的json tag 和注释都可以写上,这个过程无聊且重复,因为在设计表结构的时候你已经写过一遍了。
type User struct {
Id int64 `json:"id"`
Name string `json:"name"`
Age int64
Ctime time.Time
Mtime time.Time // 更新时间
}
定义好结构体,我们写一个查询年龄在20以下且按照id字段顺序排序的前20名用户的 go代码
func FindUsers(ctx context.Context) ([]*User, error) {
rows, err := db.QueryContext(ctx, "SELECT `id`,`name`,`age`,`ctime`,`mtime` FROM user WHERE `age`<? ORDER BY `id` LIMIT 20 ", 20)
if err != nil {
return nil, err
}
defer rows.Close()
result := []*User{}
for rows.Next() {
a := &User{}
if err := rows.Scan(&a.Id, &a.Name, &a.Age, &a.Ctime, &a.Mtime); err != nil {
return nil, err
}
result = append(result, a)
}
if rows.Err() != nil {
return nil, rows.Err()
}
return result, nil
}
当我们写少量这样的代码的时候我们可能还觉得轻松,但是当你业务工期排的很紧,并且要写大量的定制化查询的时候,这样的重复代码会越来越多。
上面的的代码我们发现有这么几个问题:
-
SQL 语句是硬编码在程序里面的,当我需要增加查询条件的时候我需要另外再写一个方法,整个方法需要拷贝一份,很不灵活。
-
在查询表所有字段的情况下,第2行下面的代码都是一样重复的,不管sql语句后面的条件是怎么样的。
-
我们发现第1行SQL语句编写和rows.Scan()那行,写的枯燥层度是和表字段的数量成正比的,如果一个表有50个字段或者100个字段,手写是非常乏味的。
-
在开发过程中rows.Close() 和 rows.Err()忘记写是常见的错误。
我们总结出来用database/sql标准库开发的痛点:
开发效率很低
很显然写上面的那种代码是很耗费时间的,因为手误容易写错,无可避免要增加自测的时间。如果上面的结构体User、 查询方法FindUsers() 代码能够自动生成,那么那将会极大的提高开发效率并且减少human error的发生从而提高开发质量。
心智负担很重
如果一个开发人员把大量的时间花在这些代码上,那么他其实是在浪费自己的时间,不管在工作中还是在个人项目中,应该把重点花在架构设计,业务逻辑设计,困难点攻坚上面,去探索和开拓自己没有经验的领域,这块Dao层的代码最好在10分钟内完成。
ORM的核心组成
明白了上面的痛点,为了开发工作更舒服,更高效,我们尝试着自己去开发一个ORM,核心的地方在于两个方面:
-
SQLBuilder:SQL语句要非硬编码,通过某种链式调用构造器帮助我构建SQL语句。
-
Scanner:从数据库返回的数据可以自动映射赋值到结构体中。
SQL SelectBuilder
我们尝试做个简略版的查询语句构造器,最终我们要达到如下图所示的效果。
我们可以通过和SQL关键字同名的方法来表达SQL语句的固有关键字,通过go方法参数来设置其中动态变化的元素,这样链式调用和写SQL语句的思维顺序是一致的,只不过我们之前通过硬编码的方式变成了方法调用。
具体代码如下:
type SelectBuilder struct {
builder *strings.Builder
column []string
tableName string
where []func(s *SelectBuilder)
args []interface{}
orderby string
offset *int64
limit *int64
}
func (s *SelectBuilder) Select(field ...string) *SelectBuilder {
s.column = append(s.column, field...)
return s
}
func (s *SelectBuilder) From(name string) *SelectBuilder {
s.tabelName = name
return s
}
func (s *SelectBuilder) Where(f ...func(s *SelectBuilder)) *SelectBuilder {
s.where = append(s.where, f...)
return s
}
func (s *SelectBuilder) OrderBy(field string) *SelectBuilder {
s.orderby = field
return s
}
func (s *SelectBuilder) Limit(offset, limit int64) *SelectBuilder {
s.offset = &offset
s.limit = &limit
return s
}
func GT(field string, arg interface{}) func(s *SelectBuilder) {
return func(s *SelectBuilder) {
s.builder.WriteString("`" + field + "`" + " > ?")
s.args = append(s.args, arg)
}
}
func (s *SelectBuilder) Query() (string, []interface{}) {
s.builder.WriteString("SELECT ")
for k, v := range s.column {
if k > 0 {
s.builder.WriteString(",")
}
s.builder.WriteString("`" + v + "`")
}
s.builder.WriteString(" FROM ")
s.builder.WriteString("`" + s.tableName + "` ")
if len(s.where) > 0 {
s.builder.WriteString("WHERE ")
for k, f := range s.where {
if k > 0 {
s.builder.WriteString(" AND ")
}
f(s)
}
}
if s.orderby != "" {
s.builder.WriteString(" ORDER BY " + s.orderby)
}
if s.limit != nil {
s.builder.WriteString(" LIMIT ")
s.builder.WriteString(strconv.FormatInt(*s.limit, 10))
}
if s.offset != nil {
s.builder.WriteString(" OFFSET ")
s.builder.WriteString(strconv.FormatInt(*s.offset, 10))
}
return s.builder.String(), s.args
}
-
通过结构体上的方法调用返回自身,使其具有链式调用能力,并通过方法调用设置结构体中的值,用以构成SQL语句需要的元素。
-
SelectBuilder 包含性能较高的strings.Builder 来拼接字符串。
-
Query()方法构建出真正的SQL语句,返回包含占位符的SQL语句和args参数。
-
[]func(s *SelectBuilder)通过函数数组来创建查询条件,可以通过函数调用的顺序和层级来生成 AND OR这种有嵌套关系的查询条件子句。
-
Where() 传入的是查询条件函数,为可变参数列表,查询条件之间默认是AND关系。
外部使用起来效果:
b := SelectBuilder{builder: &strings.Builder{}}
sql, args := b.
Select("id", "name", "age", "ctime", "mtime").
From("user").
Where(GT("id", 0), GT("age", 0)).
OrderBy("id").
Limit(0, 20).
Query()
Scanner的实现
顾名思义Scanner的作用就是把查询结果设置到对应的go对象上去,完成关系和对象的映射,关键核心就是通过反射获知传入对象的类型和字段类型,通过反射创建对象和值,并通过golang结构体的字段后面的tag来和查询结果的表头一一对应,达到动态给结构字段赋值的能力。
具体实现如下:
func ScanSlice(rows *sql.Rows, dst interface{}) error {
defer rows.Close()
// dst的地址
val := reflect.ValueOf(dst) // &[]*main.User
// 判断是否是指针类型,go是值传递,只有传指针才能让更改生效
if val.Kind() != reflect.Ptr {
return errors.New("dst not a pointer")
}
// 指针指向的Value
val = reflect.Indirect(val) // []*main.User
if val.Kind() != reflect.Slice {
return errors.New("dst not a pointer to slice")
}
// 获取slice中的类型
struPointer := val.Type().Elem() // *main.User
// 指针指向的类型 具体结构体
stru := struPointer.Elem() // main.User
cols, err := rows.Columns() // [id,name,age,ctime,mtime]
if err != nil {
return err
}
// 判断查询的字段数是否大于 结构体的字段数
if stru.NumField() < len(cols) { // 5,5
return errors.New("NumField and cols not match")
}
//结构体的json tag的value对应字段在结构体中的index
tagIdx := make(map[string]int) //map tag -> field idx
for i := 0; i < stru.NumField(); i++ {
tagname := stru.Field(i).Tag.Get("json")
if tagname != "" {
tagIdx[tagname] = i
}
}
resultType := make([]reflect.Type, 0, len(cols)) // [int64,string,int64,time.Time,time.Time]
index := make([]int, 0, len(cols)) // [0,1,2,3,4,5]
// 查找和列名相对应的结构体jsontag name的字段类型,保存类型和序号到resultType和index中
for _, v := range cols {
if i, ok := tagIdx[v]; ok {
resultType = append(resultType, stru.Field(i).Type)
index = append(index, i)
}
}
for rows.Next() {
// 创建结构体指针,获取指针指向的对象
obj := reflect.New(stru).Elem() // main.User
result := make([]interface{}, 0, len(resultType)) //[]
// 创建结构体字段类型实例的指针,并转化为interface{} 类型
for _, v := range resultType {
result = append(result, reflect.New(v).Interface()) // *Int64 ,*string ....
}
// 扫描结果
err := rows.Scan(result...)
if err != nil {
return err
}
for i, v := range result {
// 找到对应的结构体index
fieldIndex := index[i]
// 把scan 后的值通过反射得到指针指向的value,赋值给对应的结构体字段
obj.Field(fieldIndex).Set(reflect.ValueOf(v).Elem()) // 给obj 的每个字段赋值
}
// append 到slice
vv := reflect.Append(val, obj.Addr()) // append到 []*main.User, maybe addr change
val.Set(vv) // []*main.User
}
return rows.Err()
}
通过反射赋值流程,如果想知道具体的实现细节可以仔细阅读上面代码里面的注释
-
以上主要的思想就是通过reflect包来获取传入dst的Slice类型,并通过反射创建其包含的对象,具体的步骤和解释请仔细阅读注释和图例。
-
通过指定的json tag 可以把查询结果和结构体字段mapping起来,即使查询语句中字段不按照表结构顺序。
-
ScanSlice是通用的Scanner。
-
使用反射创建对象明显创建了多余的对象,没有传统的方式赋值高效,但是换来的巨大的灵活性在某些场景下是值得的。
有了SQLBuilder和Scanner 我们就可以这样写查询函数了:
func FindUserReflect() ([]*User, error) {
b := SelectBuilder{builder: &strings.Builder{}}
sql, args := b.
Select("id", "name", "age", "ctime", "mtime").
From("user").
Where(GT("id", 0), GT("age", 0)).
OrderBy("id").
Limit(0, 20).
Query()
rows, err := db.QueryContext(ctx, sql, args...)
if err != nil {
return nil, err
}
result := []*User{}
err = ScanSlice(rows, &result)
if err != nil {
return nil, err
}
return result, nil
}
生成的查询SQL语句和args如下:
SELECT `id`,`name`,`age`,`ctime`,`mtime` FROM `user` WHERE `id` > ? AND `age` > ? ORDER BY id LIMIT 20 OFFSET 0 [0 0]
自动生成
通过上面的使用的例子来看,我们的工作轻松了不少:
-
第一:SQL语句不需要硬编码了;
-
第二:Scan不需要写大量结构体字段和的乏味的重复代码。
着实帮我们省了很大的麻烦。但是查询字段还需要我们自己手写,像这种
Select("id", "name", "age", "ctime", "mtime").
-
其中传入的字段需要我们硬编码,我们可不可以再进一步,通过表结构定义来生成我们的golang结构体呢?答案是肯定的,要实现这一步我们需要一个SQL语句的解析器(https://github.com/xwb1989/sqlparser),把SQL DDL语句解析成go语言中如下的Table对象,其所包含的表名,列名、列类型、注释等都能获取到,再通过这些对象和写好的模板代码来生成我们实际业务使用的代码。
Table对象如下:
type Table struct {
TableName string // table name
GoTableName string // go struct name
PackageName string // package name
Fields []*Column // columns
}
type Column struct {
ColumnName string // column_name
ColumnType string // column_type
ColumnComment string // column_comment
}
使用以上Table对象的模板代码:
type {{.GoTableName}} struct {
{{- range .Fields }}
{{ .GoColumnName }} {{ .GoColumnType }} `json:"{{ .ColumnName }}"` // {{ .ColumnComment }}
{{- end}}
}
const (
table = "{{.TableName}}"
{{- range .Fields}}
{{ .GoColumnName}} = "{{.ColumnName}}"
{{- end }}
)
var columns = []string{
{{- range .Fields}}
{{ .GoColumnName}},
{{- end }}
}
通过上面的模板我们用user表的建表SQL语句生成如下代码:
type User struct {
Id int64 `json:"id"` // id字段
Name string `json:"name"` // 名称
Age int64 `json:"age"` // 年龄
Ctime time.Time `json:"ctime"` // 创建时间
Mtime time.Time `json:"mtime"` // 更新时间
}
const (
table = "user"
Id = "id"
Name = "name"
Age = "age"
Ctime = "ctime"
Mtime = "mtime"
)
var Columns = []string{"id","name","age","ctime","mtime"}
那么我们在查询的时候就可以这样使用
Select(Columns...)
通过模板自动生成代码,可以大大的减轻开发编码负担,使我们从繁重的代码中解放出来。
reflect真的有必要吗?
由于我们SELECT时选择查找的字段和顺序是不固定的,我们有可能 SELECT id, name, age FROM user,也可能 SELECT name, id FROM user,有很大的任意性,这种情况使用反射出来的结构体tag和查询的列名来确定映射关系是必须的。但是有一种情况我们不需要用到反射,而且是一种最常用的情况,即:查询的字段名和表结构的列名一致,且顺序一致。这时候我们可以这么写,通过DeepEqual来判断查询字段和表结构字段是否一致且顺序一致来决定是否通过反射还是通过传统方法来创建对象。用传统方式创建对象(如下图第12行)令我们编码痛苦,不过可以通过模板来自动生成下面的代码,以避免手写,这样既灵活方便好用,性能又没有损耗,看起来是一个比较完美的解决方案。
func FindUserNoReflect(b *SelectBuilder) ([]*User, error) {
sql, args := b.Query()
rows, err := db.QueryContext(ctx, sql, args...)
if err != nil {
return nil, err
}
result := []*User{}
if DeepEqual(b.column, Columns) {
defer rows.Close()
for rows.Next() {
a := &User{}
if err := rows.Scan(&a.Id, &a.Name, &a.Age, &a.Ctime, &a.Mtime); err != nil {
return nil, err
}
result = append(result, a)
}
if rows.Err() != nil {
return nil, rows.Err()
}
return result, nil
}
err = ScanSlice(rows, &result)
if err != nil {
return nil, err
}
return result, nil
}
总结
-
通过database/sql 库开发有较大痛点,ORM就是为了解决以上问题而生,其存在是有意义的。
-
ORM 两个关键的部分是SQLBuilder和Scanner的实现。
-
ORM Scanner 使用反射创建对象在性能上肯定会有一定的损失,但是带来极大的灵活性,同时在查询全表字段这种特殊情况下规避使用反射来提高性能。
展望
通过表结构,我们可以生成对应的结构体和持久层增删改查代码,我们再往前扩展一步,能否通过表结构生成的proto格式的message,以及一些常用的CRUD GRPC rpc接口定义。通过工具,我们甚至可以把前端的代码都生成好,实现半自动化编程。我想这个是值得期待的。
参考资料
[1] https://github.com/ent/ent
标签:rows,return,err,nil,如何,ORM,go,SelectBuilder,id From: https://www.cnblogs.com/rsapaper/p/17145060.html