• 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 //引用分支
    
    • 1
    • 2

    但是不管是引用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
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    针对分支或commit升降级(前提是知道之前一个commit的伪版本信息)

    比如你之前一个引用的commit是

    github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
    
    • 1

    目前是:

    github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
    
    • 1

    想要切换回去,可能go get就不是很管用了,那么就需要使用新的方式:

    replace github.com/coreos/go-systemd => github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf
    
    • 1

    即可将依赖的库指定到想要的commit

    PS:伪版本的概念

    这种是因为依赖模块的代码仓库上不存在任何tag,所以go get 默认拉取的是master分支最新一次commit对应版本的代码,并且在go.mod文件里为模块分配格式为

    v0.0.0-[主干分支最新一次commit的时间]-[commit哈希值]
    
    • 1

    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 ...")
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    运行结果大家猜能不能打印出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 ...")
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    优化点:

    • 注意打印异常日志
    • 使用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)
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    输出结果:

    [a b c]
    0xc000004078
    [a b c]
    0xc0000040a8
    
    • 1
    • 2
    • 3
    • 4

    今天的分享就先到这里咯~

  • 相关阅读:
    05_对象性能模式
    Linux安装nginx详细步骤
    ubuntu虚拟机终端使用安装脚本报错,请问如何解决
    java的IO流-字符流
    ohos的代码同步以及添加自己的代码
    【Netty 从成神到升仙系列 三】Netty 凭什么成为国内最流行的网络通信框架?
    ITSS认证系统运维范围定义
    UE 材质编辑器快捷键
    WPF-封装自定义雷达图控件
    1-丁基咪唑四氟硼酸盐([HC4im]BF4)|1-丁基-3-甲基咪唑四氟硼酸盐(BMI-BF4)|1-甲基3-丁基咪唑六氟磷酸盐([C4mim]PF6)
  • 原文地址:https://blog.csdn.net/Mr_YanMingXin/article/details/127708555