Go语言开发小技巧&易错点100例(二)

简介: 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

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

相关文章
|
14小时前
|
存储 编译器 Go
Go语言学习12-数据的使用
【5月更文挑战第5天】本篇 Huazie 向大家介绍 Go 语言数据的使用,包含赋值语句、常量与变量、可比性与有序性
16 6
Go语言学习12-数据的使用
|
1天前
|
Java Go
一文带你速通go语言指针
Go语言指针入门指南:简述指针用于提升效率,通过地址操作变量。文章作者sharkChili是Java/CSDN专家,维护Java Guide项目。文中介绍指针声明、取值,展示如何通过指针修改变量值及在函数中的应用。通过实例解析如何使用指针优化函数,以实现对原变量的直接修改。作者还邀请读者加入交流群深入探讨,并鼓励关注其公众号“写代码的SharkChili”。
8 0
|
1天前
|
存储 缓存 Java
来聊聊go语言的hashMap
本文介绍了Go语言中的`map`与Java的不同设计思想。作者`sharkChili`是一名Java和Go开发者,同时也是CSDN博客专家及JavaGuide项目的维护者。文章探讨了Go语言`map`的数据结构,包括`count`、`buckets指针`和`bmap`,解释了键值对的存储方式,如何利用内存对齐优化空间使用,并展示了`map`的初始化、插入键值对以及查找数据的源码过程。此外,作者还分享了如何通过汇编查看`map`操作,并鼓励读者深入研究Go的哈希冲突解决和源码。最后,作者提供了一个交流群,供读者讨论相关话题。
9 0
|
2天前
|
Java Go
Go语言学习11-数据初始化
【5月更文挑战第3天】本篇带大家通过内建函数 new 和 make 了解Go语言的数据初始化过程
16 1
Go语言学习11-数据初始化
|
3天前
|
自然语言处理 安全 Java
速通Go语言编译过程
Go语言编译过程详解:从词法分析(生成token)到句法分析(构建语法树),再到语义分析(类型检查、推断、匹配及函数内联)、生成中间码(SSA)和汇编码。最后,通过链接生成可执行文件。作者sharkchili,CSDN Java博客专家,分享技术细节,邀请读者加入交流群。
21 2
|
3天前
|
Java Linux Go
一文带你速通Go语言基础语法
本文是关于Go语言的入门介绍,作者因其简洁高效的特性对Go语言情有独钟。文章首先概述了Go语言的优势,包括快速上手、并发编程简单、设计简洁且功能强大,以及丰富的标准库。接着,文章通过示例展示了如何编写和运行Go代码,包括声明包、导入包和输出语句。此外,还介绍了Go的语法基础,如变量类型(数字、字符串、布尔和复数)、变量赋值、类型转换和默认值。文章还涉及条件分支(if和switch)和循环结构(for)。最后,简要提到了Go函数的定义和多返回值特性,以及一些常见的Go命令。作者计划在后续文章中进一步探讨Go语言的其他方面。
10 0
|
4天前
|
JavaScript 前端开发 Go
Go语言的入门学习
【4月更文挑战第7天】Go语言,通常称为Golang,是由Google设计并开发的一种编程语言,它于2009年公开发布。Go的设计团队主要包括Robert Griesemer、Rob Pike和Ken Thompson,这三位都是计算机科学和软件工程领域的杰出人物。
12 1
|
4天前
|
Go
|
5天前
|
分布式计算 Java Go
Golang深入浅出之-Go语言中的分布式计算框架Apache Beam
【5月更文挑战第6天】Apache Beam是一个统一的编程模型,适用于批处理和流处理,主要支持Java和Python,但也提供实验性的Go SDK。Go SDK的基本概念包括`PTransform`、`PCollection`和`Pipeline`。在使用中,需注意类型转换、窗口和触发器配置、资源管理和错误处理。尽管Go SDK文档有限,生态系统尚不成熟,且性能可能不高,但它仍为分布式计算提供了可移植的解决方案。通过理解和掌握Beam模型,开发者能编写高效的数据处理程序。
134 1
|
5天前
|
算法 关系型数据库 MySQL
Go语言中的分布式ID生成器设计与实现
【5月更文挑战第6天】本文探讨了Go语言在分布式系统中生成全局唯一ID的策略,包括Twitter的Snowflake算法、UUID和MySQL自增ID。Snowflake算法通过时间戳、节点ID和序列号生成ID,Go实现中需处理时间回拨问题。UUID保证全局唯一,但长度较长。MySQL自增ID依赖数据库,可能造成性能瓶颈。选择策略时需考虑业务需求和并发、时间同步等挑战,以确保系统稳定可靠。
112 0