Go批量操作excel导入到mongodb的技巧
怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《Go批量操作excel导入到mongodb的技巧》,涉及到Excel、导入、mongo,有需要的可以收藏一下
需求:完成一个命令工具,批量处理某个目录下面的一些excel,将这些excel数据导入到mongodb,同时可以同步到mysql
:: 花了一天时间写完代码,代码库位置:https://gitee.com/foz/lib/tree/master/ecc
代码目录:
├─cmd | └─ecc.go # 命令 ├─configs ├─data ├─internal │ └─importing # 主要逻辑处理 ├─pkg # 处理文件读取、连接数据库等 │ ├─files │ ├─mongo │ └─mysql ├─queue └─tools
1. 选择命令行包
平常使用的的命令工具包有:
- urfave/cli
- spf13/cobra
这里使用的是urfave/cli包,比较简单
var DirPath = "../data" // 默认位置
var dir = DirPath
app := &cli.App{
Name: "Ecc",
Usage: "Ecc is a tools for batch processing of excel data",
Flags: []cli.Flag{
&cli.StringFlag{
Name: "model",
Aliases: []string{"m"},
Usage: "The model of searching",
Value: "model",
Destination: &model,
},
&cli.StringFlag{ // 设置一个 -d 的参数,用来确定目标文件夹位置
Name: "dir",
Aliases: []string{"d"},
Usage: "Folder location of data files",
Destination: &dir,
Value: DirPath,
},
Action: func(c *cli.Context) error {
importing.Load("../configs/cfg.yaml") // 引入配置文件,读取mongodb、mysql等配置
importing.Handle(dir) ## 具体逻辑处理
return nil
}
2. 读取配置,连接数据库
读取配置使用spf13/viper库,需要读取一下配置,连接mongodb
var C Config
type Config struct {
Env string `yaml:"env"`
Mongo struct {
DNS string `yaml:"dns"`
Db string `yaml:"db"`
Collection string `yaml:"collection"`
} `yaml:"mongo"`
Mysql struct {
Alias string `yaml:"alias"`
Dns string `yaml:"dns"`
} `yaml:"mysql"`
}
func Load(cf string) {
var err error
viper.SetConfigFile(cf)
if err = viper.ReadInConfig(); err != nil {
log.Fatal(fmt.Errorf("fatal error config file: %s \n", err))
}
if err = viper.Unmarshal(&configs.C); err != nil {
log.Fatal(fmt.Errorf("unmarshal conf failed, err:%s \n", err))
if err = mongo.Conn(configs.C.Mongo.DNS, configs.C.Mongo.Db); err != nil {
log.Fatal(color.RedString("%s:\n%v", "mongo connect err", err))
if mongo.CheckCollection(configs.C.Mongo.Collection) {
if err = mongo.DelCollection(configs.C.Mongo.Collection); err != nil {
log.Fatal(color.RedString("%s:\n%v", "mongo del collection err", err))
}
if err = mongo.CreateCollection(configs.C.Mongo.Collection); err != nil {
log.Fatal(color.RedString("%s:\n%v", "mongo create collection err", err))
3. 读取文件
先确定文件权限以及文件是否存在
func ReadDir(dir string) ([]os.FileInfo, error) {
perm := checkPermission(dir)
if perm == true {
return nil, fmt.Errorf("permission denied dir: %s", dir)
}
if isNotExistDir(dir) {
return nil, fmt.Errorf("does not exist dir: %s", dir)
files, err := ioutil.ReadDir(dir)
if err == nil {
return files, err
return nil, fmt.Errorf("ReadDir: %s, err: %v", dir, err)
}
拿到文件后就要并发读取每个excel文件数据
这里需求是一次任务必须读完所有的文件,任何一个文件有错误就退出程序。
:: 所以需要定义异常退出信道和一个完成读取两个信道,总的数据使用sync.Map安全并发写入。
3.1. 并发读
rWait = true
rDone = make(chan struct{})
rCrash = make(chan struct{})
read(f, dir, data)
for rWait { // 使用for循环来阻塞读文件
select {
case Failure")
return
case
3.2. 使用excelize处理excel
excelize是一个非常好用的excel处理库,这里使用这个库读取excel文件内容
type ExcelPre struct {
FileName string
Data [][]string
Fields []string
Prefixes string
ProgressBar *mpb.Bar // 进度条对象
}
func ReadExcel(filePath, fileName string, pb *mpb.Progress) (err error, pre *ExcelPre) {
f, err := excelize.OpenFile(filePath)
if err != nil {
return err, nil
}
defer func() {
if _e := f.Close(); _e != nil {
fmt.Printf("%s: %v.\n\n", filePath, _e)
}
}()
// 获取第一页数据
firstSheet := f.WorkBook.Sheets.Sheet[0].Name
rows, err := f.GetRows(firstSheet)
lRows := len(rows)
if lRows
3.3. 使用mpb在命令行输出进度显示
mpb是一个很好用的命令行进度输出库,上面代码里里有两个进度条,一个是读进度条,第二个是写进度条,读进度条在文件读取的时候就显示了,返回的结构体里有写进度条对象,便于后面写操作时候显示。
下面是两个进度条显示的配置,具体参数可以看这个库的文档。
func ReadBar(total int, name string, pb *mpb.Progress) *mpb.Bar {
return pb.AddBar(int64(total),
mpb.PrependDecorators(
decor.OnComplete(decor.Name(color.YellowString("reading"), decor.WCSyncSpaceR), color.YellowString("waiting")),
decor.CountersNoUnit("%d / %d", decor.WCSyncWidth, decor.WCSyncSpaceR),
),
mpb.AppendDecorators(
decor.NewPercentage("%.2f:", decor.WCSyncSpaceR),
decor.EwmaETA(decor.ET_STYLE_MMSS, 0, decor.WCSyncWidth),
decor.Name(": "+name),
)
}
func WriteBar(total int, name string, beforeBar *mpb.Bar, pb *mpb.Progress) *mpb.Bar {
mpb.BarQueueAfter(beforeBar, false),
mpb.BarFillerClearOnComplete(),
decor.OnComplete(decor.Name(color.YellowString("writing"), decor.WCSyncSpaceR), color.GreenString("done")),
decor.OnComplete(decor.CountersNoUnit("%d / %d", decor.WCSyncSpaceR), ""),
decor.OnComplete(decor.NewPercentage("%.2f:", decor.WCSyncSpaceR), ""),
decor.OnComplete(decor.EwmaETA(decor.ET_STYLE_MMSS, 0, decor.WCSyncWidth), ""),
decor.OnComplete(decor.Name(": "+name), name),
4. 写入mongodb
同写入操作,这里拿到所有数据,然后使用goroutine并发写入mongodb,在处理数据时候需要查重,还需要记录一下本次操作插入了哪些数据的_id值,在报错的时候进行删除(这里可以使用事务,直接删除简单些),所以定义了一个Shuttle结构体用来在记录并发时的数据。
wWait = true
wDone = make(chan struct{})
wCrash = make(chan struct{})
type Shuttle struct {
Hid []string // 用来判断是否是重复数据
Mid []string // 用来记录本次插入的数据_id
mu sync.Mutex
}
func (s *Shuttle) Append(t string, str string) {
s.mu.Lock()
defer s.mu.Unlock()
switch t {
case "h":
s.Hid = append(s.Hid, str)
case "m":
s.Mid = append(s.Mid, str)
}
write2mongo(data)
for wWait {
select {
case Failure")
return
case
5. 同步mysql
因为同步mysql不是必要的,这里使用命令行输入进行判断:
tools.Yellow("-> Whether to sync data to mysql? (y/n)")
if !tools.Scan("aborted") {
return
} else {
tools.Yellow("-> Syncing data to mysql...")
if err = write2mysql(); err != nil {
tools.Red("-> Failure:" + err.Error())
} else {
tools.Green("-> Success.")
}
}
连接mysql数据库,拿到当前monogodb的数据:
func write2mysql() error {
if err := mysql.Conn(configs.C.Mysql.Dns); err != nil {
return err
}
d, err := mongo.GetCollectionAllData(configs.C.Mongo.Collection)
if err != nil {
err = Write2Mysql(d)
return err
}
创建表,直接拼sql就完事了:
func CreateTable(tableName string, fields []string) error {
var err error
delSql := fmt.Sprintf("DROP TABLE IF EXISTS `%s`", tableName)
err = Db.Exec(delSql).Error
if err != nil {
return err
}
s := "id bigint(20) NOT NULL PRIMARY KEY"
for _, field := range fields {
s += fmt.Sprintf(",%s varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL", field)
sql := fmt.Sprintf("CREATE TABLE `%s` (%s) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci", tableName, s)
err = Db.Exec(sql).Error
return nil
}
插入数据,bson.M本身就是一个map,转一下使用gorm分批插入数据,速度快一点:
func InsertData(tableName string, fields []string, data []bson.M) error {
var err error
var maps []map[string]interface{}
for _, d := range data {
row := make(map[string]interface{})
for _, field := range fields {
row[field] = d[field]
}
if row != nil {
row["id"] = d["id"].(string)
maps = append(maps, row)
}
if len(maps) > 0 {
err = Db.Table(tableName).CreateInBatches(maps, 100).Error
if err != nil {
return err
return err
}
6. 总结
做为golang新手,看了很多文档、文章,好似懂了,其实啥都不懂。
理论要掌握,实操不能落!以上关于《Go批量操作excel导入到mongodb的技巧》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
go语言实现两个协程交替打印
- 上一篇
- go语言实现两个协程交替打印
- 下一篇
- 从零开始学Golang的接口
-
- 内向的斑马
- 这篇文章内容真及时,很详细,很棒,已加入收藏夹了,关注老哥了!希望老哥能多写Golang相关的文章。
- 2023-02-12 05:32:29
-
- 大力的柚子
- 这篇博文太及时了,太详细了,感谢大佬分享,码起来,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-02-04 17:37:41
-
- 淡定的缘分
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢老哥分享技术贴!
- 2023-01-31 09:47:26
-
- 悲凉的人生
- 这篇技术贴出现的刚刚好,楼主加油!
- 2023-01-02 22:54:21
-
- 洁净的可乐
- 太详细了,收藏了,感谢up主的这篇技术贴,我会继续支持!
- 2022-12-29 08:33:54
-
- 傻傻的玫瑰
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢大佬分享文章!
- 2022-12-26 06:32:21
-
- 心灵美的小懒虫
- 这篇技术文章太及时了,太全面了,受益颇多,已收藏,关注老哥了!希望老哥能多写Golang相关的文章。
- 2022-12-24 07:42:38
-
- Golang · Go教程 | 16小时前 | goroutine · Context · 超时控制 · Go教程 · 后端开发 · Go Goroutine context 超时控制 WithTimeout Done QueryContext
- Go context 超时控制实战:从接口入口到 goroutine 回收的完整流程
- 166浏览 收藏
-
- Golang · Go教程 | 2天前 | map · 并发安全 · RWMutex · sync.Map · Go教程 · 并发安全 RWMutex sync.Map Go map并发读写 go test race
- Go map 并发读写崩溃怎么办:从复现报错到 RWMutex 修复的完整流程
- 272浏览 收藏
-
- Golang · Go教程 | 4天前 | singleflight · 并发控制 · Go教程 · 缓存治理 · 接口优化 · Go 并发请求 缓存击穿 singleflight 缓存回填
- Go singleflight 防缓存击穿实战:相同请求只查一次数据库
- 114浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 534次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 547次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 505次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 681次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 662次使用
-
- GoJava算法之Excel表列名称示例详解
- 2023-01-23 190浏览
-
- Go结合Gin导出Mysql数据到Excel表格
- 2023-01-01 352浏览
-
- golang实现浏览器导出excel文件功能
- 2022-12-31 224浏览
-
- 手把手教你导入Go语言第三方库
- 2023-01-01 312浏览
-
- Golang生成Excel文档的方法步骤
- 2022-12-31 387浏览

