Gorm的使用心得和一些常用扩展

Gorm是golang的一个orm框架,它提供了对数据库操作的封装,使用起来相当便利。

但在项目开发中,代码写的多了,还是发现在它之上还是有再次封装的空间,比如说添加错误日志、或者是一些使用频率非常高的对单个表的条件查询、分页查询、数据更新等。再则是,关于相同的功能操作,gorm也提供多种实现方式,对新学多少有些困惑,不知道该用哪个好。

于是,我基于自己在项目中的使用经验和编码习惯,做了如下一些扩展,供大家参考。

准备

为了兼容gorm的使用方法,我使用了内嵌类型来扩展。 定义如下:

type DBExtension struct {
    *gorm.DB
    logger DBLogger
}

这样子定义的wrapper对象是最小侵入式的扩展,不仅可以直接点出gorm的原有方法,也可以点出扩展的方法。

新增

关于新建数据,我建议使用Save方法,当匹配主键的数据不存在时,它的效果是插入一条新数据,而当匹配主键的数据存在时,则更新全部字段,再说一遍,它会更新全部字段

无论字段是否做了修改或者是否是定义类型的默认值

请再次注意:默认值是否生效在gorm的不同方法中处理的方式是不一样的,需要非常小心才行。

举个例子,如果你定义了一个User的结构体,里面有个Age的字段类型是int。(注:以后的例子,都默认已定义这个结构体

type User struct {
    Id           int     `gorm:"column:id; type:int(11);primary_key"`
    Name         string  `gorm:"column:name; type:varchar(32);"`
    Age          int     `gorm:"column:age; type:int(11);"`
    Description  string  `gorm:"column:description; type:varchar(512);"`
}

func (User) TableName() string {
    return "test.user"
}

++请特别注意Id的定义中的primary_key, 如果没有加个这个Save方法是无法正常工作的。++

如果在定义时,没有给Age赋值,那么这条数据的Age将被置为0。

对于新增数据,可能问题不大,但是对于数据更新,那这就可就是一个隐晦的bug了!

那既然Save方法有这样一个坑,为什么还要用它呢?

简单来说,不用显示的判断是新增数据和更新数据,可以让代码更加简洁,利大于弊,不是吗?

扩展代码如下,增加了一些错误判断和日志:

type TableNameAble interface {
    TableName() string
}

// Update All Fields
func (dw *DBExtension) SaveOne(value TableNameAble) error {
    tableNameAble, ok := value.(TableNameAble)
    if !ok {
        return errors.New("value doesn't implement TableNameAble")
    }

    var err error
    if err = dw.Save(value).Error; err != nil {
        dw.logger.LogErrorc("mysql", err, fmt.Sprintf("Failed to save %s, the value is %+v", tableNameAble.TableName(), value))
    }
    return err
}

使用代码如下:


user1 := User{Id:1, Name:"Jeremy", Age: 30, Description: "A gopher"}

if err := dw.SaveOne(&instInfo); err != nil{
    // error handling
    return err
}

当记录不存在时,执行的Sql语句是:

insert into test.user(id ,name, age, description) values(1, "Jeremy", 30, "A gopher")

当记录存在时,执行的语句就是:

update test.user set name = "Jeremy", age = 30, description = "A gohper" where id = 1

这样写新建,还兼顾了全字段更新的情况,是不是一举两得呢?

PS: 如果主键Id是一个自增列,在新建时,可以不用给Id赋值。当数据成功插入后,这条数据的Id还会自动更新到Id字段,这个特性在一些场景下特别有用。

更新

SaveOne方法是全量更新,但大部分情况是,可能只是更新某条数据的部分字段,又或者是只想更新改过的字段。关于这部分操作,gorm虽然提供了很多操作方法,但也是最让人困惑的。

在这种场景我常用的处理方式有两种,一是定义一个专门的结构体,如:

type UserDesc struct {
    Id           int     `gorm:"column:id; type:int(11);primary_key"`
    Description  string  `gorm:"column:description; type:varchar(512);"`
}

func (UserDesc) TableName() string {
    return "test.user"
}

这时就可以使用SaveOne方法,用如下方式更新:


userDesc := UserDesc{Id:1,  Description: "A programmer"}

if err := dw.SaveOne(&userDesc); err != nil{
    // error handling
    return err
}

执行的sql语句是:

update test.user set description = "A programmer" where id = 1

但是更多的时候,是想按匹配条件更新的匹配的数据,这时SaveOne就无法满足了。于是,我做了如下扩展:

const table_name =  "$Table_Name$"

type UpdateAttrs map[string]interface{}

func NewUpdateAttrs(tableName string) UpdateAttrs  {
    attrMap := make(map[string]interface{})
    attrMap[table_name] = tableName
    return attrMap
}

// Update selected Fields, if attrs is an object, it will ignore default value field; if attrs is map, it will ignore unchanged field.
func (dw *DBExtension) Update(attrs interface{}, query interface{}, args ...interface{}) error {
    var (
        tableNameAble TableNameAble
        ok            bool
        tableName     string
    )

    if tableNameAble, ok = query.(TableNameAble); ok {
        tableName = tableNameAble.TableName()
    }else if tableNameAble, ok = attrs.(TableNameAble); ok {
        tableName = tableNameAble.TableName()
    } else if attrMap, isUpdateAttrs := attrs.(UpdateAttrs); isUpdateAttrs {
        tableName = attrMap[table_name].(string)
        delete(attrMap, table_name)
    }

    if tableName == "" {
        return errors.New("can't get table name from both attrs and query")
    }

    var err error
    db := dw.Table(tableName).Where(query, args...).Update(attrs)

    if err = db.Error; err != nil {
        dw.logger.LogErrorc("mysql", err, fmt.Sprintf("failed to update %s, query is %+v, args are %+v, attrs is %+v", tableName, query, args, attrs))
    }

    if db.RowsAffected == 0 {
        dw.logger.LogWarnc("mysql",nil, fmt.Sprintf("No rows is updated.For %s, query is %+v, args are %+v, attrs is %+v", tableName, query, args, attrs))
    }

    return err
}

下面,我将结合Sql语句,逐一解释如何使用。

还是先以要执行下面这条语句为例:

update test.user set description = "A programmer" where id = 1

现在,可以有如下几种实现方式

  • 写法一
udateAttrs := User{Description: "A programmer"}
condition := User{Id: 1}
if err := dw.Update(&udateAttrs, condition); err != nil{
    // error handling
    return err
}
  • 写法二
udateAttrs := User{Description: "A programmer"}
if err := dw.Update(&udateAttrs, "id = ?", 1); err != nil{
    // error handling
    return err
}
  • 写法三
udateAttrs := NewUpdateAttrs("test.user")
udateAttrs["description"] = "A programmer"

if err := dw.Update(&udateAttrs, "id = ?", 1); err != nil{
    // error handling
    return err
}
  • 写法四
udateAttrs := NewUpdateAttrs("test.user")
udateAttrs["description"] = "A programmer"
condition := User{Id: 1}

if err := dw.Update(&udateAttrs, condition); err != nil{
    // error handling
    return err
}

咋一看,四种写法很相似。那么,为什么要搞这么多种写法呢?

这可是不是为了炫耀回字的几种写法, 而是因为gorm原生的Update方法对于struct的参数是会忽略默认值的。

比如说,如果你想把descritpion清空,如果像下面这样写:

udateAttrs := User{Description: ""}
condition := User{Id: 1}
if err := dw.Update(&udateAttrs, condition); err != nil{
    // error handling
    return err
}

descritpion是不会被更新的,这是就需要写法三或者写法四了,以写法四为例

udateAttrs := NewUpdateAttrs("test.user")
udateAttrs["description"] = ""
condition := User{Id: 1}
if err := dw.Update(&udateAttrs, condition); err != nil{
    // error handling
    return err
}

才会如愿执行:

update test.user set description = "" where id = 1

而写法二(三)的强大之处在于可以更自由的指定匹配条件,比如:

udateAttrs := User{Description: "A programmer"}
if err := dw.Update(&udateAttrs, "id in (?) and age > ? and description != ?", []int{1,2}, 30, ""); err != nil{
    // error handling
    return err
}

执行的sql是:

update test.user set description = "A programmer" where id in (1,2) and age > 30 and description != ''

未完待续……

代码地址:Github:Ksloveyuan/gorm-ex

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容