一文秒懂Go 编写命令行工具的代码
亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《一文秒懂Go 编写命令行工具的代码》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下命令行、工具,希望所有认真读完的童鞋们,都有实质性的提高。
但现阶段相对来说还是 Python
写的多一些,偶尔还得回炉写点 Java
;自然对 Go
也谈不上多熟悉。
于是便利用周末时间自己做个小项目来加深一些使用经验。于是我便想到了之前利用 Java
写的一个博客小工具。
那段时间正值微博图床大量图片禁止外链,导致许多个人博客中的图片都不能查看。这个工具可以将文章中的图片备份到本地,还能将图片直接替换到其他图床。
我个人现在是一直在使用,通常是在码字的时候利用 iPic
之类的工具将图片上传到微博图床(主要是方便+免费)。写完之后再通过这个工具一键切换到 [SM.MS](http://sm.MS)
这类付费图床,同时也会将图片备份到本地磁盘。
改为用 Go
重写为 cli
工具后使用效果如下:
需要掌握哪些技能
之所以选择这个工具用 Go
来重写;一个是功能比较简单,但也正好可以利用到 Go
的一些特点,比如网络 IO、协程同步之类。
同时修改为命令行工具后是不是感觉更极客了呢。
再开始之前还是先为不熟悉 Go
的 Javaer
介绍下大概会用到哪些知识点:
- 使用和管理第三方依赖包(
go mod
) - 协程的运用。
- 多平台打包。
下面开始具体操作,我觉得即便是没怎么接触过 Go
的朋友看完之后也能快速上手实现一个小工具。
使用和管理第三方依赖
- 还没有安装 Go 的朋友请参考官网自行安装。
首先介绍一下 Go 的依赖管理,在版本 1.11
之后官方就自带了依赖管理模块,所以在当下最新版 1.15
中已经强烈推荐使用。
它的目的和作用与 Java
中的 maven
,Python
中的 pip
类似,但使用起来比 maven
简单许多。
根据它的使用参考,需要首先在项目目录下执行 go mod init
用于初始化一个 go.mod
文件,当然如果你使用的是 GoLang
这样的 IDE
,在新建项目时会自动帮我们创建好目录结构,当然也包含 go.mod
这个文件。
在这个文件中我们引入我们需要的第三方包:
module btb go 1.15 require ( github.com/cheggaaa/pb/v3 v3.0.5 github.com/fatih/color v1.10.0 github.com/urfave/cli/v2 v2.3.0 )
我这里使用了三个包,分别是:
pb
: progress bar,用于在控制台输出进度条。color
: 用于在控制台输出不同颜色的文本。cli
: 命令行工具开发包。
import ( "btb/constants" "btb/service" "github.com/urfave/cli/v2" "log" "os" ) func main() { var model string downloadPath := constants.DownloadPath markdownPath := constants.MarkdownPath app := &cli.App{ Flags: []cli.Flag{ &cli.StringFlag{ Name: "model", Usage: "operating mode; r:replace, b:backup", DefaultText: "b", Aliases: []string{"m"}, Required: true, Destination: &model, }, &cli.StringFlag{ Name: "download-path", Usage: "The path where the image is stored", Aliases: []string{"dp"}, Destination: &downloadPath, Required: true, Value: constants.DownloadPath, }, &cli.StringFlag{ Name: "markdown-path", Usage: "The path where the markdown file is stored", Aliases: []string{"mp"}, Destination: &markdownPath, Required: true, Value: constants.MarkdownPath, }, }, Action: func(c *cli.Context) error { service.DownLoadPic(markdownPath, downloadPath) return nil }, Name: "btb", Usage: "Help you backup and replace your blog's images", } err := app.Run(os.Args) if err != nil { log.Fatal(err) } }
代码非常简单,无非就是使用了 cli
所提供的 api 创建了几个命令,将用户输入的 -dp
、-mp
参数映射到 downloadPath
、markdownPath
变量中。
之后便利用这两个数据扫描所有的图片,以及将图片下载到对应的目录中。
更多使用指南可以直接参考官方文档。
可以看到部分语法与 Java
完全不同,比如:
- 申明变量时类型是放在后边,先定义变量名称;方法参数类似。
- 类型推导,可以不指定变量类型(新版本的
Java
也支持)方法支持同时返回多个值,这点非常好用。 - 公共、私用函数利用首字母大小写来区分。
- 还有其他的就不一一列举了。
协程
紧接着命令执行处调用了 service.DownLoadPic(markdownPath, downloadPath)
处理业务逻辑。
这里包含的文件扫描、图片下载之类的代码就不分析了;官方 SDK
写的很清楚,也比较简单。
重点看看 Go
里的 goroutime
也就是协程。
我这里使用的场景是每扫描到一个文件就利用一个协程去解析和下载图片,从而可以提高整体的运行效率。
func DownLoadPic(markdownPath, downloadPath string) { wg := sync.WaitGroup{} allFile, err := util.GetAllFile(markdownPath) wg.Add(len(*allFile)) if err != nil { log.Fatal("read file error") } for _, filePath := range *allFile { go func(filePath string) { allLine, err := util.ReadFileLine(filePath) if err != nil { log.Fatal(err) } availableImgs := util.MatchAvailableImg(allLine) bar := pb.ProgressBarTemplate(constants.PbTmpl).Start(len(*availableImgs)) bar.Set("fileName", filePath). SetWidth(120) for _, url := range *availableImgs { if err != nil { log.Fatal(err) } err := util.DownloadFile(url, *genFullFileName(downloadPath, filePath, &url)) if err != nil { log.Fatal(err) } bar.Increment() } bar.Finish() wg.Done() }(filePath) } wg.Wait() color.Green("Successful handling of [%v] files.\n", len(*allFile)) if err != nil { log.Fatal(err) } }
就代码使用层面看起来是不是要比 Java
简洁许多,我们不用像 Java
那样需要维护一个 executorService
,也不需要考虑这个线程池的大小,一切都交给 Go
自己去调度。
使用时只需要在调用函数之前加上 go
关键字,只不过这里是一个匿名函数。
而且由于 goroutime
非常轻量,与 Java
中的 thread
相比占用非常少的内存,所以我们也不需要精准的控制创建数量。
不过这里也用到了一个和 Java
非常类似的东西:WaitGroup
。
它的用法与作用都与 Java
中的 CountDownLatch
非常相似;主要用于等待所有的 goroutime
执行完毕,在这里自然是等待所有的图片都下载完毕然后退出程序。
使用起来主要分为三步:
- 创建和初始化
goruntime
的数量:wg.Add(len(number)
- 每当一个
goruntime
执行完毕调用wg.Done()
让计数减一。 - 最终调用
wg.Wait()
等待WaitGroup
的数量减为0。
对于协程 Go 推荐使用 chanel
来互相通信,这点今后有机会再讨论。
打包
核心逻辑也就这么多,下面来讲讲打包与运行;这点和 Java
的区别就比较大了。
众所周知,Java
有一句名言:write once run anywhere
这是因为有了 JVM
虚拟机,所以我们不管代码最终运行于哪个平台都只需要打出一个包;但 Go
没有虚拟机它是怎么做到在个各平台运行呢。
简单来说 Go
可以针对不同平台打包出不同的二进制文件,这个文件包含了所有运行所需要的依赖,甚至都不需要在目标平台安装 Go
环境。
- 虽说 Java 最终只需要打一个包,但也得在各个平台安装兼容的
Java
运行环境。
我在这里编写了一个 Makefile
用于执行打包:make release
# Binary name BINARY=btb GOBUILD=go build -ldflags "-s -w" -o ${BINARY} GOCLEAN=go clean RMTARGZ=rm -rf *.gz VERSION=0.0.1 release: # Clean $(GOCLEAN) $(RMTARGZ) # Build for mac CGO_ENABLED=0 GOOS=darwin GOARCH=amd64 $(GOBUILD) tar czvf ${BINARY}-mac64-${VERSION}.tar.gz ./${BINARY} # Build for arm $(GOCLEAN) CGO_ENABLED=0 GOOS=linux GOARCH=arm64 $(GOBUILD) tar czvf ${BINARY}-arm64-${VERSION}.tar.gz ./${BINARY} # Build for linux $(GOCLEAN) CGO_ENABLED=0 GOOS=linux GOARCH=amd64 $(GOBUILD) tar czvf ${BINARY}-linux64-${VERSION}.tar.gz ./${BINARY} # Build for win $(GOCLEAN) CGO_ENABLED=0 GOOS=windows GOARCH=amd64 $(GOBUILD).exe tar czvf ${BINARY}-win64-${VERSION}.tar.gz ./${BINARY}.exe $(GOCLEAN)
可以看到我们只需要在 go build
之前指定系统变量即可打出不同平台的包,比如我们为 Linux
系统的 arm64
架构打包文件:
CGO_ENABLED=0 GOOS=linux GOARCH=arm64 go build main.go -o btb
便可以直接在目标平台执行 ./btb
运行程序。
总结
本文所有代码都已上传 Github
: https://github.com/crossoverJie/btb
感兴趣的也可以直接运行安装脚本体验。
curl -fsSL https://raw.githubusercontent.com/crossoverJie/btb/master/install.sh | bash
目前这个版本只实现了图片下载备份,后续会完善图床替换及其他功能。
这段时间接触 Go
之后给我的感触颇深,对于年纪 25 岁的 Java
来说,Go
确实是后生可畏,更气人的是还赶上了云原生这个浪潮,就更惹不起了。
一些以前看来不那么重要的小毛病也被重点放大,比如启动慢、占用内存多、语法啰嗦等;不过我依然对这位赏饭吃的祖师爷保持期待,从新版本的 Java
可以看出也在积极改变,更不用说它还有无人撼动的庞大生态。
文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《一文秒懂Go 编写命令行工具的代码》文章吧,也可关注golang学习网公众号了解相关技术文章。

- 上一篇
- Golang实现Directional Channel(定向通道)

- 下一篇
- 模块一 GO语言基础知识-库源码文件
-
- 快乐的萝莉
- 这篇文章出现的刚刚好,好细啊,感谢大佬分享,已加入收藏夹了,关注老哥了!希望老哥能多写Golang相关的文章。
- 2023-03-08 16:54:09
-
- 狂野的帆布鞋
- 这篇文章出现的刚刚好,太细致了,很好,码住,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-02-13 13:34:45
-
- 火星上的凉面
- 太全面了,码住,感谢博主的这篇博文,我会继续支持!
- 2023-02-07 11:51:52
-
- 甜美的心锁
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢up主分享技术贴!
- 2023-01-23 10:29:17
-
- 舒服的春天
- 这篇技术文章真及时,太详细了,太给力了,已加入收藏夹了,关注楼主了!希望楼主能多写Golang相关的文章。
- 2023-01-07 00:26:04
-
- 沉默的小刺猬
- 太细致了,收藏了,感谢大佬的这篇博文,我会继续支持!
- 2023-01-04 20:37:59
-
- 甜美的睫毛膏
- 这篇博文真是及时雨啊,太详细了,写的不错,已收藏,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-01-02 03:23:15
-
- 淡然的蜜粉
- 写的不错,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢博主分享文章内容!
- 2023-01-02 03:03:11
-
- 害羞的玫瑰
- 这篇技术文章真是及时雨啊,师傅加油!
- 2022-12-31 06:39:43
-
- Golang · Go教程 | 1天前 |
- TigervncDebian多用户共享桌面超简单教程
- 482浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Go语言新手必看!切片vs数组,一次搞定这两个核心知识点
- 472浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Docker在Debian上运行超简单教程(保姆级教学)
- 210浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian设置hostname踩坑记录:权限问题大揭秘
- 334浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian装SQLServer?这些问题你一定要注意!
- 284浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian系统下Jenkins自动化部署脚本教学
- 367浏览 收藏
-
- Golang · Go教程 | 2天前 |
- DebianSwap服务器应用实测,这些场景真的用得上!
- 319浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian跑TigerVNC实测!真香警告,快来看看性能咋样~
- 171浏览 收藏
-
- Golang · Go教程 | 2天前 |
- 在Debian上玩转SQLServer备份还原,手把手教你一步步操作
- 498浏览 收藏
-
- Golang · Go教程 | 2天前 |
- DebianOverlay不会玩?手把手教你轻松定制化安装
- 258浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Go语言实战:time.Ticker&time.After用法区别及避坑技巧
- 240浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian系统如何快速定位&干掉那些讨厌的僵尸进程
- 317浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 茅茅虫AIGC检测
- 茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
- 21次使用
-
- 赛林匹克平台(Challympics)
- 探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
- 50次使用
-
- 笔格AIPPT
- SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
- 58次使用
-
- 稿定PPT
- 告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
- 53次使用
-
- Suno苏诺中文版
- 探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
- 60次使用
-
- Golang操作命令行的几种方式总结
- 2022-12-27 426浏览
-
- Golang执行cmd命令行的方法
- 2023-01-20 412浏览
-
- golang执行命令行的实现
- 2023-01-07 109浏览
-
- Go代码检查的推荐工具及使用详解
- 2023-01-07 491浏览
-
- Go语言包管理工具Godep的用法
- 2022-12-31 302浏览