本期看点(技巧类用【技】表示,易错点用【易】表示):
(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
今天的分享就先到这里咯~
转自:https://juejin.cn/post/7172174161861672997
最后编辑: kuteng 文档更新时间: 2024-03-27 11:19 作者:kuteng