【GoLang】【框架学习】【GORM】4. 使用 BeforeUpdate hook 操作时,出现反射报错

文章目录

  • [0. 问题背景](#0. 问题背景)
  • [1. 问题环境](#1. 问题环境)
  • [2. 问题排查](#2. 问题排查)
  • [3. 总结](#3. 总结)

0. 问题背景

对于大多的业务操作的增删查改来说,都会记录一个 创建者、更新者。我们的业务上是通过 GORM hook 的方式来进行处理的。在 create 的时候用的挺顺的,但在 update 的时候出现反射的报错,最终分析代码后,可能是我们使用方式有误,或者 GORM 本身不支持该操作导致的。

在 Github 上找到了同样遇见此问题的 issue,但没有解决,我在下面也做了相应的评论:

1. 问题环境

对业务做了简单抽象,后 ORM 结构定义:

go 复制代码
type User struct {
    BaseModelOper
    Name  string
    Age   int
    Email string
}

func (u *User) TableName() string {
    return "user"
}

type BaseModelOper struct {
    ID        int64          `gorm:"primary_key;auto_increment:true;type:bigint" json:"id"`
    CreatedAt time.Time      `json:"created_at" gorm:"type:timestamp;autoCreateTime:milli"`
    UpdatedAt time.Time      `json:"updated_at" gorm:"type:timestamp;autoUpdateTime:milli"`
    DeletedAt gorm.DeletedAt `json:"deleted_at" gorm:"type:timestamp"`
    CreatedBy int64          `json:"created_by" gorm:"type:bigint"`
    UpdatedBy int64          `json:"updated_by" gorm:"type:bigint"`
}

func (*BaseModelOper) BeforeCreate(db *gorm.DB) error {
    userId := GetUserIdFromCtx(db.Statement.Context)
    if userId != 0 {
        db.Statement.SetColumn("created_by", userId)
    }
    return nil
}

func (*BaseModelOper) BeforeUpdate(db *gorm.DB) error {
    userId := GetUserIdFromCtx(db.Statement.Context)
    if userId != 0 {
        db.Statement.SetColumn("updated_by", userId)
    }
    return nil
}

update 操作:

go 复制代码
// 查询条件
type UserCond struct {
    Id *int64 `json:"id" gorm:"type:bigint"`
}
// 更新结构
type UserUpdate struct {
    Name *string
}

func (u *User) UpdateByCond(ctx context.Context, cond UserCond, updateV *UserUpdate) error {
    query, values := util.MakeStructQuery(cond)
    err = dbcon.CtxDB(ctx).Model(&User{}).Where(query, values...).Updates(updateV).Error
    if err != nil {
        logger.Errorf(ctx, "UpdateByCond has err. %+v", err)
        return err
    }
    return nil
}

建表、数据插入、数据更新操作:

go 复制代码
func TestUpdateBy() {
    if err := dbcon.GetDB().AutoMigrate(&User{}); err != nil {
        panic(err)
    }
    ctx := context.Background()
    ctx = SetUserIdToCtx(1)
    u := User{
        Age:   21,
        Name:  "AAA",
        Email: "xrc@xrc.com111",
    }
    if err := dbcon.CtxDB(ctx).Create(&u).Error; err != nil {
        panic(err)
    }
    
    if err := u.UpdateByCond(
        ctx,
        UserCond{Id: proto.Int64(1)},
        &UserUpdate{Name: proto.String("BBB")}); err != nil {
        panic(err)
    }
}

出现了 panic 错误!

2. 问题排查

  • 实际上就是设置单列的更新的操作。
  • 有几个关键信息点需要先了解:
    • stmt.Dest 就是我们传递进来的精简结构体 UserUpdate。其中不包含 UpdatedBy 这些基础字段,仅包含业务字段。
    • stmt.Schema 是 ORM 结构体,也就是对数据表映射的结构体。包含全部字段,即 基础字段+业务字段。

流程如下:

  • 注意上面的 field.Set 传入的是 destValue,这里的 destValue 字段是 ORM 结构的精简版,并不包含全体字段。

  • 这里的 field 是 ORM 结构体的结构,是 0,5

    • 0 指的是 ORM 结构体中第一个匿名结构体
    • 5 是这个匿名结构体中的第 5 个字段
    • 也就是说,需要通过反射,找到结构体里面的 [0, 5] 字段,给他赋值。

  • 但是这里的 v 是 dest 反射值的解引用,实际上是一个不完整的结构,能看到里面只有一个字段。

  • 所以当 i=5 的时候,在 dest 中就找不到对应的字段,那么自然就会报错。

3. 总结

正如我在上面 issue 的评论一样:

这块没明白为何要这样要求?在 SetColumn 的操作里面,dest 如果是结构体类型的话,就需要使用表对应的结构体类型才行。针对 create 类的操作确实没啥问题。而 update 操作既然已经支持了 struct/map 两种更新方式,为何在这又不支持自定义 struct 更新了呢?

涉及到一些代码逻辑的历史问题,简单的将自定义结构体通过序列化转换成了 map 结构才得以暂时解决...

既然 update 操作支持通过 struct 的方式去做更新,为何不支持的更彻底一点???先如今用 map 的方式去转换处理,代码属实冗余。

相关推荐
ZIM学编程4 小时前
「学长有话说」作为一个大三学长,我想对大一计算机专业学生说这些!
java·c语言·数据结构·c++·python·学习·php
maizeman1264 小时前
用R语言生成指定品种与对照的一元回归直线(含置信区间)
开发语言·回归·r语言·置信区间·品种测试
脚踏实地的大梦想家5 小时前
【Go】P17 Go语言并发编程核心:深入理解 Goroutine (从入门到实战)
java·开发语言·golang
初学小白...5 小时前
线程同步机制及三大不安全案例
java·开发语言·jvm
Felicity_Gao5 小时前
uni-app App升级功能实现
前端·学习·uni-app
CS Beginner5 小时前
【搭建】个人博客网站的搭建
java·前端·学习·servlet·log4j·mybatis
reept6 小时前
Pytorch常用函数学习摘录
人工智能·pytorch·学习
用坏多个鼠标6 小时前
Nacos和Nginx集群,项目启动失败问题
java·开发语言
满天星83035776 小时前
【C++】右值引用和移动语义
开发语言·c++·redis·visual studio