当前位置:首页 > 文章列表 > Golang > Go问答 > 在gorm中执行批量更新操作

在gorm中执行批量更新操作

来源:stackoverflow 2024-03-09 19:27:22 0浏览 收藏

本篇文章向大家介绍《在gorm中执行批量更新操作》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。

问题内容

我有一个需求,更新bulk中的数据,而这个数据中的一些字段经过二次处理后,变成了不同的。比如

[{"id":83,"ip":"10.215.14.216","test1":24,"test2":126,"test3":300},
{"id":82,"ip":"10.215.14.215","test1":6,"test2":100,"test3":600}
...]

官方gorm批量更新是将部分字段改为相同的值

最简单的方法是使用loop一次更新一个,不过个人感觉in高效

for _,v := range bulkdata{
  // update single
}

有没有批量更新的方法来提高效率?我真的很感谢任何对此的帮助。


正确答案


  1. 当 类似时尝试使用 case
update table_name
set ip = case id
    when 83 then '10.215.14.216'
    when 82 then '10.215.14.215'
end, test1 = case id
    when 83 then 24,
    when 82 then 6
end
where id in (83, 82)
  • 现在使用 reflect 生成原始 sql
  • package main
    
    import (
        "bytes"
        "fmt"
        "math"
        "reflect"
        "strconv"
        "strings"
    )
    
    func genBatchUpdateSQL(tableName string, dataList interface{}) ([]string, error) {
        fieldValue := reflect.ValueOf(dataList)
        fieldType := reflect.TypeOf(dataList).Elem().Elem()
        sliceLength := fieldValue.Len()
        fieldNum := fieldType.NumField()
         
        // TODO add validate struct. struct must have primaryKey and gorm tag
    
        var IDList []string
        updateMap := make(map[string][]string)
        for i := 0; i < sliceLength; i++ {
            structValue := fieldValue.Index(i).Elem()
            for j := 0; j < fieldNum; j++ {
                elem := structValue.Field(j)
    
                var tid string
                switch elem.Kind() {
                case reflect.Int64:
                    tid = strconv.FormatInt(elem.Int(), 10)
                case reflect.String:
                    if strings.Contains(elem.String(), "'") {
                        tid = fmt.Sprintf("'%v'", strings.ReplaceAll(elem.String(), "'", "\\'"))
                    } else {
                        tid = fmt.Sprintf("'%v'", elem.String())
                    }
                case reflect.Float64:
                    tid = strconv.FormatFloat(elem.Float(), 'f', -1, 64)
                case reflect.Bool:
                    tid = strconv.FormatBool(elem.Bool())
                default:
                    return nil, fmt.Errorf("type conversion error, param is %v", fieldType.Field(j).Tag.Get("json"))
                }
    
                gormTag := fieldType.Field(j).Tag.Get("gorm")
                fieldTag := getFieldName(gormTag)
    
                if strings.HasPrefix(fieldTag, "id;") {
                    id, err := strconv.ParseInt(tid, 10, 64)
                    if err != nil {
                        return nil, err
                    }
    
                    if id < 1 {
                        return nil, fmt.Errorf("this structure should have a primary key and gt 0")
                    }
                    IDList = append(IDList, tid)
                    continue
                }
    
                valueList := append(updateMap[fieldTag], tid)
                updateMap[fieldTag] = valueList
            }
        }
    
        length := len(IDList)
        // Length of each batch submission
        size := 200
        SQLQuantity := getSQLQuantity(length, size)
        var SQLArray []string
        k := 0
    
        for i := 0; i < SQLQuantity; i++ {
            count := 0
            var record bytes.Buffer
            record.WriteString("UPDATE " + tableName + " SET ")
    
            for fieldName, fieldValueList := range updateMap {
                record.WriteString(fieldName)
                record.WriteString(" = CASE " + "id")
    
                for j := k; j < len(IDList) && j < len(fieldValueList) && j < size+k; j++ {
                    record.WriteString(" WHEN " + IDList[j] + " THEN " + fieldValueList[j])
                }
    
                count++
                if count != fieldNum-1 {
                    record.WriteString(" END, ")
                }
            }
    
            record.WriteString(" END WHERE ")
            record.WriteString("id" + " IN (")
            min := size + k
            if len(IDList) < min {
                min = len(IDList)
            }
            record.WriteString(strings.Join(IDList[k:min], ","))
            record.WriteString(");")
    
            k += size
            SQLArray = append(SQLArray, record.String())
        }
        return SQLArray, nil
    }
    
    func getSQLQuantity(length, size int) int {
        return int(math.Ceil(float64(length) / float64(size)))
    }
    
    func getFieldName(fieldTag string) string {
        fieldTagArr := strings.Split(fieldTag, ":")
        if len(fieldTagArr) == 0 {
            return ""
        }
    
        return fieldTagArr[len(fieldTagArr)-1]
    }
    
    
    type Ts struct {
        ID    int64   `gorm:"column:id;primaryKey" json:"id"`
        IP    string  `gorm:"column:ip" json:"ip"`
        Test1 float64 `gorm:"column:test1" json:"test1"`
    }
    
    func main() {
        var t []*Ts
        demo1 := &Ts{1, "11.215.14.216", 11.0}
        demo2 := &Ts{2, "12.215.14.216", 12.0}
        demo3 := &Ts{3, "13.215.14.216", 13.0}
        demo4 := &Ts{4, "14.215.14.216", 14.0}
        demo5 := &Ts{5, "15.215.14.216", 15.0}
    
        t = append(t, demo1, demo2, demo3, demo4, demo5)
        res, err := genBatchUpdateSQL("table_name", t)
        if err != nil {
            return
        }
        fmt.Println(res)
        // [UPDATE table_name SET ip = CASE id WHEN 1 THEN 'Test001' WHEN 2 THEN 'Test002' WHEN 3 THEN 'Test003' WHEN 4 THEN 'Test004' WHEN 5 THEN 'Test005' END, test1 = CASE id WHEN 1 THEN 11 WHEN 2 THEN 12 WHEN 3 THEN 13 WHEN 4 THEN 14 WHEN 5 THEN 15 END WHERE id IN (1,2,3,4,5);]
    }
    
    
    1. 现在您可以使用原始 sql 进行批量更新

    好了,本文到此结束,带大家了解了《在gorm中执行批量更新操作》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

    版本声明
    本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
    解决Discuz删除回复的常见问题解决Discuz删除回复的常见问题
    上一篇
    解决Discuz删除回复的常见问题
    使用 JSON 格式传递消息通过 PubSub 实现
    下一篇
    使用 JSON 格式传递消息通过 PubSub 实现
    查看更多
    最新文章
    查看更多
    课程推荐
    • 前端进阶之JavaScript设计模式
      前端进阶之JavaScript设计模式
      设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
      542次学习
    • GO语言核心编程课程
      GO语言核心编程课程
      本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
      508次学习
    • 简单聊聊mysql8与网络通信
      简单聊聊mysql8与网络通信
      如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
      497次学习
    • JavaScript正则表达式基础与实战
      JavaScript正则表达式基础与实战
      在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
      487次学习
    • 从零制作响应式网站—Grid布局
      从零制作响应式网站—Grid布局
      本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
      484次学习
    查看更多
    AI推荐
    • 美图AI抠图:行业领先的智能图像处理技术,3秒出图,精准无误
      美图AI抠图
      美图AI抠图,依托CVPR 2024竞赛亚军技术,提供顶尖的图像处理解决方案。适用于证件照、商品、毛发等多场景,支持批量处理,3秒出图,零PS基础也能轻松操作,满足个人与商业需求。
      5次使用
    • SEO标题PetGPT:智能桌面宠物程序,结合AI对话的个性化陪伴工具
      PetGPT
      SEO摘要PetGPT 是一款基于 Python 和 PyQt 开发的智能桌面宠物程序,集成了 OpenAI 的 GPT 模型,提供上下文感知对话和主动聊天功能。用户可高度自定义宠物的外观和行为,支持插件热更新和二次开发。适用于需要陪伴和效率辅助的办公族、学生及 AI 技术爱好者。
      5次使用
    • 可图AI图片生成:快手可灵AI2.0引领图像创作新时代
      可图AI图片生成
      探索快手旗下可灵AI2.0发布的可图AI2.0图像生成大模型,体验从文本生成图像、图像编辑到风格转绘的全链路创作。了解其技术突破、功能创新及在广告、影视、非遗等领域的应用,领先于Midjourney、DALL-E等竞品。
      41次使用
    • MeowTalk喵说:AI猫咪语言翻译,增进人猫情感交流
      MeowTalk喵说
      MeowTalk喵说是一款由Akvelon公司开发的AI应用,通过分析猫咪的叫声,帮助主人理解猫咪的需求和情感。支持iOS和Android平台,提供个性化翻译、情感互动、趣味对话等功能,增进人猫之间的情感联系。
      35次使用
    • SEO标题Traini:全球首创宠物AI技术,提升宠物健康与行为解读
      Traini
      SEO摘要Traini是一家专注于宠物健康教育的创新科技公司,利用先进的人工智能技术,提供宠物行为解读、个性化训练计划、在线课程、医疗辅助和个性化服务推荐等多功能服务。通过PEBI系统,Traini能够精准识别宠物狗的12种情绪状态,推动宠物与人类的智能互动,提升宠物生活质量。
      35次使用
    微信登录更方便
    • 密码登录
    • 注册账号
    登录即同意 用户协议隐私政策
    返回登录
    • 重置密码