往期回顾:
本期看点(技巧类用【技】表示,易错点用【易】表示):
(1)Go Module中对依赖库版本的升级与降级【技】
(2)Go goroutine中的异常处理【易】
(3)Go中slice作为参数是值传递【技】
正文:
1 Go Module中对依赖库版本的升级与降级【技】
Go Module引用其他的依赖库分为两种方式:
- 引用该库的某一个tag
- 引用该库的某一个分支(称之为伪版本)
因而显示上也会有差别(大多数情况下),如下:
github.com/coreos/go-semver v0.3.0 //引用tag github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf //引用分支
但是不管是引用tag还是引用分支,指向的都是该仓库的分支或tag提交最后一个commit时的镜像,因而版本的升级与降级的最小颗粒度就是commit
先看针对tag进行升降级:
# 查看该依赖库的所有版本 go list -m -versions github.com/xuri/excelize/v2 # 输出: github.com/xuri/excelize/v2 v2.0.0 v2.0.1 v2.0.2 v2.1.0 v2.2.0 v2.3.0 v2.3.1 v2.3.2 v2.4.0 v2.4.1 v2.5.0 v2.6.0 v2.6.1 # 查看当前项目依赖库的版本 go list -m github.com/xuri/excelize/v2 # 输出: github.com/xuri/excelize/v2 v2.6.1 # 指定想要切换到的版本 go get github.com/xuri/excelize/v2@v2.6.0 # 输出: go: downloading github.com/xuri/excelize/v2 v2.6.0 go get: downgraded github.com/xuri/excelize/v2 v2.6.1 => v2.6.0
针对分支或commit升降级(前提是知道之前一个commit的伪版本信息):
比如你之前一个引用的commit是
github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
目前是:
github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
想要切换回去,可能go get就不是很管用了,那么就需要使用新的方式:
replace github.com/coreos/go-systemd => github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
即可将依赖的库指定到想要的commit
PS:伪版本的概念
这种是因为依赖模块的代码仓库上不存在任何tag,所以go get
默认拉取的是master分支最新一次commit对应版本的代码,并且在go.mod
文件里为模块分配格式为
v0.0.0-[主干分支最新一次commit的时间]-[commit哈希值]
2 Go goroutine中的异常处理【易】
首先看一下这种情况:
func main() { waitGroup := sync.WaitGroup{} waitGroup.Add(1) go func() { defer waitGroup.Done() fmt.Println("Hello goroutine ...") panic("err") }() waitGroup.Wait() fmt.Println("Hello main ...") }
运行结果大家猜能不能打印出Hello main …?
答案:不能
问题产生的原因就是在goroutine中我们没有进行对可能发生的异常进行处理,当goroutine中发生panic的时候会影响主函数的运行,那么我们应该如何处理呢?
使用recover处理异常
func main() { waitGroup := sync.WaitGroup{} waitGroup.Add(1) go func() { defer func() { if e := recover(); e != nil { fmt.Println("recover panic") } waitGroup.Done() }() fmt.Println("Hello goroutine ...") panic("err") }() waitGroup.Wait() fmt.Println("Hello main ...") }
优化点:
- 注意打印异常日志
- 使用channel将错误信息返回至主线程处理
3 Go中slice作为参数是值传递【技】
我们直接来试一下:
func Test(list []string) { fmt.Println(list) fmt.Println(unsafe.Pointer(&list)) } func main() { list := make([]string, 0) list = append(list, "a", "b", "c") fmt.Println(list) fmt.Println(unsafe.Pointer(&list)) Test(list) }
输出结果:
[a b c] 0xc000004078 [a b c] 0xc0000040a8
今天的分享就先到这里咯~