【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 的方式去转换处理,代码属实冗余。

相关推荐
jllllyuz几秒前
MATLAB实现蜻蜓优化算法
开发语言·算法·matlab
冰暮流星15 分钟前
javascript逻辑运算符
开发语言·javascript·ecmascript
flysh0516 分钟前
如何利用 C# 内置的 Action 和 Func 委托
开发语言·c#
码农小韩38 分钟前
基于Linux的C++学习——动态数组容器vector
linux·c语言·开发语言·数据结构·c++·单片机·学习
木风小助理38 分钟前
`mapfile`命令详解:Bash中高效的文本至数组转换工具
开发语言·chrome·bash
yyy(十一月限定版)1 小时前
初始matlab
开发语言·matlab
LawrenceLan1 小时前
Flutter 零基础入门(九):构造函数、命名构造函数与 this 关键字
开发语言·flutter·dart
listhi5201 小时前
基于MATLAB的支持向量机(SVM)医学图像分割方法
开发语言·matlab
嵌入式-老费1 小时前
外壳3D结构设计(学习的方法)
学习
hui函数1 小时前
如何解决 pip install 编译报错 g++: command not found(缺少 C++ 编译器)问题
开发语言·c++·pip