首页 > 其他分享 > 一次折腾 Golang 泛型、反射和 gorm 框架的记录

一次折腾 Golang 泛型、反射和 gorm 框架的记录

时间:2023-05-29 15:45:31浏览次数:62  
标签:reflect return string TableName stmt Golang 泛型 gorm

事情的起初是一个很常见的需求:批量更新多条记录的相同字段,每条记录对应的字段值不同因此无法批量 Update。看着没啥难度却没想到从开头到结束整整花了一天的时间,遂有此文。

首先尝试了 gorm 自带的 Save(),按理说 gorm 本身会自动识别零值不去更新,这样直接创建一个实例数组挨个赋值后调 Save() 就可以了,如:

data := make([]records, 0, len(n))
for i:=0; i!=len(data); i++ {
    data[i].column = values[i]
}
return db.Save(data).Error

现实很骨感:Save() 只支持类似思路的单个字段更新,敢往里塞数组这货会直接批量创建记录……

重新整理了一下思路,想到可以手搓 SQL 用 UPDATECASE WHEN THEN 语法来批量更新,控制一下每次更新的记录数量就行了。

翻了半天 gorm 浅显易懂的文档压根没有关于CASE WHEN THEN的内容,讲真要不是 gorm 有 Exec 方法我都想考虑换 ORM……其实早就想换了

实现时候又碰见了两个问题,一个是使用UPDATE的话需要往 SQL 中传入表名,如何获取表名需要一个通用的函数(不能指望每个表都手动实现TableName()),于是又花时间实现了一个通用的函数。具体内容足够再水一篇。 另外就是需要一个通用的将数字转换为字符串的函数,把结果最后拼接成 SQL。

SQL

麻烦的点主要是需要识别每个新值的类型,如果是字符串或者Time的话需要往 SQL 里塞引号,于是写了个根据反射判断类型并转为字符串的函数:

这里要注意一下不能简单的用%v%f去格式化打印浮点型,因为结果可能是科学记数法。

func ParseNumberToString(n any) (result string, ok bool) {
	var integer, decimal32, decimal64 bool
	switch n.(type) {
	// common
	case int:
		integer = true
	// ...
	case float32:
		decimal32 = true
	// ...

	if integer {
		return fmt.Sprintf("%d", n), true
	}
	if decimal32 {
		decimalValue, _ := n.(float32)
		return strconv.FormatFloat(float64(decimalValue), 'f', -1, 32), true
	}
	if decimal64 {
		decimalValue, _ := n.(float64)
		return strconv.FormatFloat(decimalValue, 'f', -1, 64), true
	}
	return "", false
}

在拼接 SQL 的时候检查是否是字符串 or Time,如果两者都不是直接调它就行了。主要内容大概这样:

query := fmt.Sprintf("UPDATE %s SET", tableName)

conditions := make([]string, 0, len(columnValuesMap))
for column, rawValues := range columnValuesMap {
	condition := fmt.Sprintf(" %s = CASE id ", column)
	args := make([]string, 0, len(ids))
	for i := 0; i != len(rawValues); i++ {
		var stringValue string
		switch rawValues[i].(type) {
		case string:
			stringValue = rawValues[i].(string)
		case time.Time:
			stringValue = rawValues[i].(time.Time).Format("2006-01-02 15:04:05")
		default:
			strValue, ok := reflects.ParseNumberToString(rawValues[i])
			// 这里不能直接写 stringValue 就极度的恶心好吧,想写还要声明一遍 ok 再把:=改成=!
			if !ok {
				return fmt.Errorf("unknown type: %T", rawValues[i])
			}
			stringValue = strValue
		}
		args = append(args, fmt.Sprintf(" WHEN %d THEN %s", ids[i], stringValue))
	}
	condition += strings.Join(args, " ")
	conditions = append(conditions, condition)
}

query += strings.Join(conditions, " END, ") + " END WHERE id IN (?)"

处理一下可能的数据不合法情况如值和 id 数量不一样,字段名为空啥的就能顺利的拼接出 SQL 了。

拼 SQL 没花多少功夫,真正费时间的在于那个生成表名的函数……

TableName

gorm 这个框架本身拥有一套自己生成表名的函数,并定义了一个Tabler接口,其中包含一个TableName() string方法来返回表名。用户可以自行实现TableName(),若未实现则会使用 gorm 自己的规则。

如果想不破坏现有的逻辑,那就只能把 gorm 自己的规则翻出来。stackoverflow 上有人也问过 类似的问题,稍加查询得到了如下代码:

func (t Table) TableName() string {
    stmt := &gorm.Statement{DB: DB}
    stmt.Parse(t)
    return stmt.Schema.Table
}

因为项目有些表已经手动实现了 TableName,所以想着不破坏现有逻辑的情况下尝试直接为每个表都会继承的基础类添上这个方法。

但陷阱在于,stmt.Parse(t) 本身也会调用 TableName,于是 test 的时候直接真的 stack overflow 了……

后来查阅了一下Parse到底在干嘛,发现还有第二种方案:

func (t Table) TableName() string {
  	stmt := &gorm.Statement{DB: DB}
    namer := stmt.NamingStrategy
    modelType := reflect.Indirect(t).Type()
    modelValue := reflect.New(modelType)
    return namer.TableName(modelType.Name())
}

完美!直接把Parse的大部分过程跳过了!

然而妄想给基础类实现这个方法会导致传入的 modelType 永远都是基础类,于是每个表的 TableName 都成了base_model……

无奈只能写一个往里传实际类型的。由于之前为了统一数据库逻辑,把所有数据库相关的表方法都改成了泛型函数,于是也要整一个泛型版出来。

func GetGormTableName[T any]() string {
	var t T
	stmt := &gorm.Statement{DB: DB}
	namer := stmt.NamingStrategy

  tType := reflect.ValueOf(t).Type()
  tTypeName := tType.Name()
	return namer.TableName(tTypeName)
}

看起来不错,唯一可惜就是不能通过方法调用。

然而又在测试时候发现了问题:泛型会直接把 T 的类型带入函数(废话),这就导致如果 T 是一个指针类型,那么var t T本质上直接创建了一个空指针,然后把这玩意给reflect.ValueOf()会直接 panic……

虽然应该没有人吃饱了撑的写GetGormTableName[*Table](),但就怕有人在今天这种日子给他打了 50。于是……

通过reflect.TypeElem()方法可以拿到指针指向的元素类型,再用reflect.New(elem)可以创建一个新的指向该类型的指针,最后用Elem().Interface()就可以获取对应的实例了。

func GetGormTableName[T any]() string {
	var t T
	stmt := &gorm.Statement{DB: global.DB}
	namer := stmt.NamingStrategy

	var tTypeName string
	if reflect.ValueOf(t).Kind() == reflect.Ptr {
		tElem := reflect.TypeOf(t).Elem()
		newPtr := reflect.New(tElem)
		tInstance := newPtr.Elem().Interface()
		tTypeName = reflect.ValueOf(tInstance).Type().Name()
	} else {
		tType := reflect.ValueOf(t).Type()
		tTypeName = tType.Name()
	}
	return namer.TableName(tTypeName)
}

顺便还写了一个判断当前类型是否拥有TableName()方法的函数,如果有的话直接返回对应的值。

func GetImplementedTableName[T any]() (result string, ok bool) {
	var t T
	var funcMayExist reflect.Value
  // 这里也涉及对于方法接收者是否为指针类型的判断,如果接收者为指针,通过对象本身调用方法也会报错,那么直接把传入的对象取地址。
	if reflect.ValueOf(t).Kind() != reflect.Ptr {
		funcMayExist = reflect.ValueOf(&t).MethodByName("TableName")
	} else {
    // 这里同样为传入的指针类型创建一个实例对象。
		tType := reflect.TypeOf(t).Elem()
		tPtr := reflect.New(tType)
		instance := tPtr.Elem().Interface()
		funcMayExist = reflect.ValueOf(instance).MethodByName("TableName")
	}
	if funcMayExist.IsValid() {
		values := funcMayExist.Call(nil)
		if len(values) != 0 {
			if values[0].Kind() == reflect.String {
				return values[0].String(), true
			}
		}
	}

	return "", false
}

忙了一天差不多算是完工了,现在 Golang 的泛型我怎么感觉怎么像是用反射换皮的……(还没换完整,不然为啥不支持结构体泛型?)

标签:reflect,return,string,TableName,stmt,Golang,泛型,gorm
From: https://www.cnblogs.com/miranquil/p/17440633.html

相关文章

  • Java中如何获得A<T>泛型中T的运行时类型及原理探究(转)
    原文:https://developer.aliyun.com/article/1226646简介如果经常写工具类,很大概率会遇到一个比较实际的问题,就是需要在泛型表达式A中获取T的运行时类型。获取它需要一些技巧。但这个技巧很少被透彻的解释过为什么会生效。在接下来的文章里,我们会从Java的泛型(Generics)谈起,结合JLS......
  • 泛型和原生类型
    //泛型类的内部类可以访问到其外围类的类型参数。//当你在一个泛型类中嵌套另一个泛型类时,最好为它们的类型参数设置不同的名字,即使那个嵌套类是静态的也应如此。classLinkedList<E>{ /*访问外围类的类型参数 privateNodehead=null; privateclassNode{ Evalue;......
  • Golang入门笔记
    第一个Go程序packagemain //包,表明代码所在的模块(包)import"fmt" //引入代码依赖//功能实现funcmain(){ fmt.Println("HelloWorld");}编译执行➜go-examplegit:(master)✗cdsrc/example/main➜maingit:(master)✗lshello_world.go➜maingit:......
  • Golang GIN 接收结构体切片(前端对象数组)
    想接收前端这样的数据格式:【json数组】[{"password":"casso","mobile":"13456789999","nick_name":"go","icon":"地址"},{"passwor......
  • Golang环境——GOPATH vs go.mod
    GOPATH在本文中,我们将探讨Go编程的传统环境与基础环境之间的差异go.mod。这种区别对Go开发人员如何构建和管理他们的项目工作区和依赖项具有重要意义。我们将从了解GOPATH环境、它的组织和结构开始。然后,我们将探讨go.mod采用这种方法来提供模块化和灵活的方式来组织Go项......
  • 泛型
    泛型类泛型T比较固定不支持继承类写入泛型?支持继承但是不可控泛型?extendsE 加上边界泛型 ?superE给泛型加下边界 泛型接口 interfacea<T,N>{}publicclassb<T,N>implements  a<T,N>{}publicclassc<String,Integer> implements  a<T,N>{} 泛......
  • [golang]gin框架接收websocket通信
    前言WebSocket是一种在单个TCP连接上进行全双工通信的协议。WebSocket让客户端和服务端之间的数据交换变得非常简单,且允许服务器主动向客户端推送数据,并且之后客户端和服务端所有的通信都依靠这个专用协议进行。本文使用gin框架编写服务端应用,配置路由接收websocket请求并处理。......
  • Day2-java泛型与集合
     ArrayList类的使用Set接口及实现类的使用了解对象顺序,Comparable接口及其使用Queue接口和实现类的使用了解Map接口及实现类的使用Collections类的实用方法操作List对象StreamAPI的使用了解泛型类型 取出和输入不用强制类型转换泛型是带一个或多个类型......
  • pongo2 类似django 语法的golang 模版引擎
    pongo2类似django语法的golang模版引擎,当前兼容django1.7同时还包含了强大的三方生态支持(比如beego,gin,echo框架。。。)参考使用main.gopackagemain import("fmt""log" "github.com/flosch/pongo2/v6") funcmain(){tpl,e......
  • golang中切片越界获取到零值的坑
    B站视频演示golang中切片越界获取到零值的坑代码截图~~~......