为什么 Go 用起来会难受?这六个坑你知道吗

开发 前端
今天我们围绕 Go 的难受场景进行了分析和讲解,本文涉及的分别是:泛型、浅拷贝和泄露、错误处理、nil 接口不是 nil、垃圾回收、依赖管理。

在做新的应用选型时,我们会进行应用编程语言的选择,这时会纠结 Java、PHP、Go...各种,会思考有没有致命的问题,不能用?

可以明确的是,Go 没有非常致命的问题,否则你我他都不会在这里相遇,也不会大火。

难受的点,倒是有不少,今天就由煎鱼和大家一起来看看。

难受的点

泛型

在 Go1.18 以前,在所有社交媒体和调查报告上来看。Go 最难受的莫过于没有泛型,

写一个通用的方法,要不得把入参声明为 interface,要不得写 N 个不同类型的一样代码的函数,代码重复率高。

如下图:

这是 Go1.18 以前最难受的点,现在新版本虽然有了泛型,但现阶段配套标准和开源库还没完全到位,影响还是会继续存在。

浅拷贝和泄露

在写 Go 程序时,我们经常要用到 slice、map 等基础类型。但有一个比较麻烦的点,就是会涉及到浅拷贝。

一个不注意就会引起 BUG,如下代码:

type T struct {
A string
B []string
}

func main() {
x := T{"煎鱼", []string{"上班"}}

y := x
y.A = "咸鱼"
y.B[0] = "下班"

fmt.Println(x)
fmt.Println(y)
}

输出结果是什么?

煎鱼到底是上班了,还是下班了?

结果如下:

{煎鱼 [下班]}
{咸鱼 [下班]}

实际上在 y := x 时,他拷贝的是指向对象的指针,这个时候 x 和 y 的底层数据其实是一家子,自然一变动 y,x 的煎鱼也就下班了。

同类型的 slice 也有 append 的泄露,以及 len、cap 的不准确问题,是比较折腾人的。

泄露的示例:

var a []int

func f(b []int) []int {
a = b[:2]
return a
}

func main() {
...
}

有兴趣的可以具体看《Go 切片导致内存泄露,被坑两次了!》的解析。

错误处理

在 Go 的错误处理中,许多小伙伴的难受的点分两大块。一个是大量重复的 if err != nil 的代码:

func main() {
x, err := foo()
if err != nil {
// handle error
}
y, err := foo()
if err != nil {
// handle error
}
z, err := foo()
if err != nil {
// handle error
}
s, err := foo()
if err != nil {
// handle error
}
}

另外一块是在异常处理中,Go 现阶段是 panic 和 recover 的模式,内部还包含 throw 的致命性错误抛出,无法拦截,为此我也见过个别事故是因此造成的。

这是一个争议性很大的板块。

nil 接口不是 nil

我们强行将一段 Go 程序的变量值赋为 nil,并进行 nil 与 nil 的判断。

代码如下:

func main() {
var v interface{}
v = (*int)(nil)
fmt.Println(v == nil)
}

输出的结果是什么。是 false,还是 true,又或是抛出异常?

输出结果是 fasle,nil 可不 100% 等于 nil。

这与 interface 的内部数据结构有关,是在编程时要注意的一个细节,具体可详见《Go 面试题:Go interface 的一个 “坑” 及原理分析》的解析。

垃圾回收

Go 非常简洁,垃圾回收唯一的可调节的是 GC 频率,可以通过 GOGC 变量设置初始垃圾收集器的目标百分比值。

$ GOGC=100 eddycjy

简单来讲就是,GOGC 的值设置的越大,GC 的频率越低,但每次最终所触发到 GC 的堆内存也会更大。

然后就没别的方式可以优化垃圾回收器本身了,以至于当年我还被人拿 Java 来吐槽过一遍,说 Go 肯定有。

依赖管理

压轴的难受点,莫过于 Go 的依赖管理。先是从 GOPATH 时代,开源后一路水土不服,后面 rsc 直接下场支棱起来硬推。

到 2022 年,目前 Go modules 还是会存在一些让人难受的点。甚至曹大总结了 《Go mod 七宗罪》,不少我在工作中也遇到和替别人解决过,非常的精辟。

引用曹大文章中的 7 点:

  • Go 命令的副作用:所有 go build、go list、go test 多多少少都会拉取到墙外的资源,会很慢。
  • 形同虚设的 semver 规范:go mod 的设计,就是希望大家在软件库发布时都要遵守标准,例如在小版本时,要保持兼容性。但这非常理想化,现实就是经常有人不遵守。
  • 无法应对删库:发布后的软件库,你已经拉取了。但发布者依然可以删除,受伤的还是自己。
  • goproxy 的实现并不统一:作者 A、作者 B、作者 C 写的几套 goproxy 内部逻辑是不完全一致的,很折腾人。
  • go get 到的 lib 版本在 go build 时被修改。
  • 版本信息扩散:导入路径是包含版本号 v1、v2 等信息的,一旦修改,就得大面积替换。
  • go.sum 合并冲突:大型项目上的多人维护,导致频繁冲突。

熟悉掌握 Go 的一个表现,那就是精通 Go modules,不然项目都运行的不顺利。

总结

今天我们围绕 Go 的难受场景进行了分析和讲解,本文涉及的分别是:泛型、浅拷贝和泄露、错误处理、nil 接口不是 nil、垃圾回收、依赖管理。

其中不少是常见的,也有的是有意而为之(例如:垃圾回收)。从大家的角度来看,你觉得 Go 比较难受的点还有哪些呢?

责任编辑:武晓燕 来源: 脑子进煎鱼了
相关推荐

2021-11-17 11:03:14

Python代码语法

2020-11-17 08:30:06

LinuxSwapping 设计

2023-11-02 10:22:29

gRPC后端通信

2024-10-09 08:19:35

2024-09-02 00:30:41

Go语言场景

2020-10-08 18:58:46

条件变量开发线程

2024-10-12 14:58:07

2020-12-24 15:26:07

Redis数据库

2022-09-14 08:11:06

分页模糊查询

2020-09-11 06:39:29

ThreadLocal线程

2021-11-10 15:37:49

Go源码指令

2022-05-09 07:49:47

PulsarJava问题排查

2021-05-31 10:22:09

Go语言代码

2022-01-05 11:40:36

Go特性语言

2024-04-30 09:02:48

2023-12-20 08:23:53

NIO组件非阻塞

2022-09-06 08:07:24

SQL语句查询

2022-11-28 00:04:17

2024-01-15 12:16:37

2024-04-23 08:31:57

pythonfalse
点赞
收藏

51CTO技术栈公众号