开启生长之旅!这是我参与「日新计划 12 月更文应战」的第7天,点击检查活动详情
往期回忆:
- Go言语开发小技巧&易错点100例(一)
本期看点(技巧类用【技】表明,易错点用【易】表明):
(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
今日的共享就先到这里咯~