跟着 Go 夜读阅读 Go 代码系列 1

我为什么会探究 iota 呢?

这就要从我最近遇到的一次 protobuf 解析报错说起,错误信息如下:

cannot parse reserved wire type

所以有关这个错误的原因,可以查看👇这篇文章

https://stackoverflow.com/questions/62033409/cannot-find-solutions-to-a-protobuf-unmarshal-error

在查看这个错误的源码定义时,发现它们是一组 const iota 定义,而 const 值是通过 -iota 来赋值的,具体代码:protobuf/encoding/protowire/wire.go

from: google.golang.org/protobuf/encoding/protowire/wire.go

1
2
3
4
5
6
7
8
const (
	_ = -iota
	errCodeTruncated
	errCodeFieldNumber
	errCodeOverflow
	errCodeReserved
	errCodeEndGroup
)

说实话,我用 Go 这么久,还是第一次见 -iota,所以我就一下子来了兴趣,特别想要了解它,首先想到的就是去查 go ref/spec https://golang.org/ref/spec#Iota,但是文中并没有提到 -iota 的用法。

为了了解清楚 iota ,所以我就开始探究,并将我探究的过程记录下来,抛砖引玉,希望可以跟大家一起探讨。

iota 由来

Iota (大写 I 小写 l,中文音译:约塔),是一个全名,而不是某一组词的缩写。 iota 是希腊字母中的第 9 个字母。

  • 希腊字母 iota 的词源 Etymology

    Etymology

    From Ancient Greek ἰῶτα (iôta).

    (jot): In reference to a phrase in the New Testament: “until heaven and earth pass away, not an iota, not a dot, will pass from the Law” (Mt 5:18), iota being the smallest letter of the Greek alphabet.

以上内容摘录自维基百科。

网上还有阐述(但是我并未能考证,保留至此,以供大家参考)

1
2
3
4
5
iota 是典型的数学符号,它所表示的含义如下:

1. 作为求和算法的迭代器
2. 作为[下标索引](https://tex.stackexchange.com/questions/96092/making-subscripts-temporarily-deeper-for-clarity) 
3. 对于[复数的虚部](https://en.wikipedia.org/wiki/Imaginary_unit#Alternative_notations)

希腊字母 iota 在编程语言 APL 中用于生成连续整数序列(可以参考阅读 iota Scheme 文献)。

1
2
3
4
APL 是 A Programming Language 或 Array Processing Language 的缩写。
肯尼斯・艾佛森在 1962 年设计这个语言时他正在哈佛大学工作,1979 年他因对数学表达式和编程语言理论的贡献而得到图灵奖。
在过去数十年的使用历史中,APL 从它的原始版本开始不断改变和发展,今天的版本与 1963 年发表时的版本已经非常不一样了。
但它始终是一种解释执行的计算机语言。

iota 也存在于 C++ 中,在其他语言中也有类似于 Go 的用法。

Go iota 究竟有何用处

可以被当做 enum 来使用,它使得我们在 const 块中,默认值为 0,即第一行为 0, 以后每一行加1;

Go 中 iota 的使用大致是基于 APL 中的定义使用。

引用 https://en.wikipedia.org/wiki/Iota

iota 的用法/注意事项/探究

  1. 不同 const 定义块互不干扰
  2. 所有注释行和空行全部忽略
  3. 从第 1 行开始,iota 从 0 逐行加 1

接下来我们来看一个 Go issues 上提到的一些有关 iota 的应用例子

以下内容来自于:https://github.com/golang/go/issues/39751

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
type myConst int

const (
        zero myConst = iota // iota = 0
        one                 // iota = 1
        three = iota + 1    // iota = 2
        foure               // iota = 3
        five = iota + 1     // iota = 4
)

func testIota() {
        // 3 4 5 why not 3 4 6
        fmt.Println(three, foure, five)
}

我们知道 iota 的值是从第一行 0 ,开始逐行递增的,对应 iota + 1 只是普通的表达式计算,对于 iota 来说是没有影响的。

我们也可以通过修改源代码,然后再编译为新的 go 来执行我们的程序,修改代码块为:src/cmd/compile/internal/noder/noder.go#L456 constDecl() 在 cs.iota++ 之前打印,得到以下值,也印证了我上面说的话。

1
2
3
4
5
constState.iota:  0
constState.iota:  1
constState.iota:  2
constState.iota:  3
constState.iota:  4

iota 能不能被用于普通变量申明?

Robert Griesemer 在 2017 年 8 月 16 日提了一个 proposal 就是想要在 Go2 做这件事。

iota 是什么❓

这个用法我们可以直接查看 Go 标准包, src/text/scanner/scanner.go:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
// The result of Scan is one of these tokens or a Unicode character.
const (
	EOF = -(iota + 1)
	Ident
	Int
	Float
	Char
	String
	RawString
	Comment

	// internal use only
	skipComment
)

如果你不通过测试,你知道 Ident 的值为 -2 吗?

还有 src/cmd/asm/internal/arch/arch.go:

1
2
3
4
5
6
7
// Pseudo-registers whose names are the constant name without the leading R.
const (
	RFP = -(iota + 1)
	RSB
	RSP
	RPC
)

以及 src/cmd/compile/internal/gc/bexport.go

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
// Tags. Must be < 0.
const (
	// Objects
	packageTag = -(iota + 1)
	constTag
	typeTag
	varTag
	funcTag
	endTag

	// Types
	namedTag
	arrayTag
	sliceTag
	dddTag
	structTag
	pointerTag
	signatureTag
	interfaceTag
	mapTag
	chanTag

	// Values
	falseTag
	trueTag
	int64Tag
	floatTag
	fractionTag // not used by gc
	complexTag
	stringTag
	nilTag
	unknownTag // not used by gc (only appears in packages with errors)

	// Type aliases
	aliasTag
)

iota 是对 iota 增加了一个负号,算是一个表达式,所以对应的结果仅仅是做了一个负值计算。

iota 源码级探究

非常感谢欧神对此处源码阅读的指导,大家如果对 Go 源码感兴趣,可以直接读欧神写的书📚《Go 语言原本

在线阅读地址:https://golang.design/under-the-hood/

为什么 -iota 之后的 const 值为递减呢?

https://golang.design/gossa?id=b49e9104-3750-4a47-ba55-e491489d8cf1

src/cmd/compile/internal/ir 下面有一个 ConstExpr

1
2
3
4
5
type ConstExpr struct {
	miniExpr
	origNode
	val constant.Value
}

注意:此处的 ConstExpr 是在 Go 仓库 master 上才有,在当前 go1.16.3 是没有这个申明的。

额外补充有关 ir 重构相关的内容

为什么要重构一个 ir 出来,我们可以根据 git commit history 追溯得知

如果要完全分解 gc 包,则需要将其定义的编译器 IR 移到一个单独的包中,该包可以由 gc 本身导入的包导入。

为什么要重构为 ConstExpr,我们可以在 CL 上看到描述:

言归正传,我们继续来看 iota 是如何计算的。

最容易想到的定位方法,就是全局检索: iota 然后查看其赋值即可,但是现实是远没有我们想象的那么简单。。。

以下截图是在重构 ConstExpr 所带来的收益,但是并不是简单做了一个 定义就可以,所以我在正式发文中将其删除了。

https://s3-us-west-2.amazonaws.com/secure.notion-static.com/3ef34fc3-656d-42b7-8205-a0c912cbfc15/Untitled.png

https://s3-us-west-2.amazonaws.com/secure.notion-static.com/1a0c6a21-b7c9-4c38-a9cf-a50c0bb16c5b/Untitled.png

https://s3-us-west-2.amazonaws.com/secure.notion-static.com/b47e9632-8a58-457c-b297-7ae0a69c34e3/Untitled.png

言归正传,我们继续来看 iota 是如何计算的。

最容易想到的定位方法,就是全局检索: iota 然后查看其赋值即可,但是现实是远没有我们想象的那么简单。。。

1
2
3
4
5
6
7
8
9
// constState tracks state between constant specifiers within a
// declaration group. This state is kept separate from noder so nested
// constant declarations are handled correctly (e.g., issue 15550).
type constState struct {
	group  *syntax.Group
	typ    ir.Ntype
	values []ir.Node
	iota   int64
}

入口函数 src/cmd/compile/main.go,

1
2
3
func main() {
	gc.Main(archInit)
}

gc.Main: src/cmd/compile/internal/gc/main.go :

1
2
3
4
5
6
func Main(archInit func(*ssagen.ArchInfo)) {  
  ...
  // Parse and typecheck input.
	noder.LoadPackage(flag.Args())
  ...
}

在 LoadPackage 主要是做了以下 5 步,详细内容如下:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
func LoadPackage(filenames []string) {
	...
	for _, p := range noders {
		p.node()
		p.file = nil // release memory
	}
	...
	// Process top-level declarations in phases.
	// Phase 1: const, type, and names and types of funcs.
	//   This will gather all the information about types
	//   and methods but doesn't depend on any of it.
	//
	//   We also defer type alias declarations until phase 2
	//   to avoid cycles like #18640.
	//   TODO(gri) Remove this again once we have a fix for #25838.

	// Don't use range--typecheck can add closures to Target.Decls.
	...
	// Phase 2: Variable assignments.
	//   To check interface assignments, depends on phase 1.

	// Don't use range--typecheck can add closures to Target.Decls.
	...
	// Phase 3: Type check function bodies.
	// Don't use range--typecheck can add closures to Target.Decls.
	...
	// Phase 4: Check external declarations.
	// TODO(mdempsky): This should be handled when type checking their
	// corresponding ODCL nodes.
	...
	// Phase 5: With all user code type-checked, it's now safe to verify map keys.
	// With all user code typechecked, it's now safe to verify unused dot imports.
}

在 Process top-level declarations in phases 之前的 p.node () 主要就干了一件事:

1
typecheck.Target.Decls = append(typecheck.Target.Decls, p.decls(p.file.DeclList)...)

对于 const 来说, decls 就是执行 func (p *noder) constDecl(decl *syntax.ConstDecl, cs *constState) []ir.Node

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
func (p *noder) constDecl(decl *syntax.ConstDecl, cs *constState) []ir.Node {
	...

	n.SetIota(cs.iota)
	...

	cs.iota++

	return nn
}

从而得到了我们的 iota 为一个递增的值。

而 typecheck 在 go 1.16.3 是在以上几个阶段的时候实时执行的,而在 master 上,已经不是这样实现了。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
func typecheck(n ir.Node, top int) (res ir.Node) {
	...
	// Resolve definition of name and value of iota lazily.
	n = Resolve(n)
	...
	n = typecheck1(n, top)
	...
	if t != nil {
		n = EvalConst(n)
		t = n.Type()
	}
	...
	return n
}

我们拆解以上逻辑块:

  1. Resolve 得到 iota 的值,注意这里的值其实是正值
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
// Resolve ONONAME to definition, if any.
func Resolve(n ir.Node) (res ir.Node) {
	...
	if r.Op() == ir.OIOTA {
	   if x := getIotaValue(); x >= 0 {
	      return ir.NewInt(x)
	   }
	   return n
	}
	...
}

getIotaValue()是从 typecheckdefstack 中取最后一个 ir.Name,然后取 Offset_ 的值。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
// type checks the whole tree of an expression.
// calculates expression types.
// evaluates compile time constants.
// marks variables that escape the local frame.
// rewrites n.Op to be more specific in some cases.

var typecheckdefstack []*ir.Name
...

// getIotaValue returns the current value for "iota",
// or -1 if not within a ConstSpec.
func getIotaValue() int64 {
	if i := len(typecheckdefstack); i > 0 {
		if x := typecheckdefstack[i-1]; x.Op() == ir.OLITERAL {
			return x.Iota()
		}
	}
}

ir.Name 中的 Offset_ 的相关操作:

1
2
func (n *Name) Iota() int64            { return n.Offset_ }
func (n *Name) SetIota(x int64)        { n.Offset_ = x }
  1. typecheck1 进行了一系列的猛操作就是为 EvalConst 铺路。

  2. EvalConst 逻辑块:

https://github.com/golang/go/blob/2ebe77a2fda1ee9ff6fd9a3e08933ad1ebaea039/src/cmd/compile/internal/typecheck/const.go#L395

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
// EvalConst returns a constant-evaluated expression equivalent to n.
// If n is not a constant, EvalConst returns n.
// Otherwise, EvalConst returns a new OLITERAL with the same value as n,
// and with .Orig pointing back to n.
func EvalConst(n ir.Node) ir.Node {
	// Pick off just the opcodes that can be constant evaluated.
	switch n.Op() {
		// 而构建 Const 的关键就是 tokenForOp[n.Op()],到底是 + 还是 -
		return OrigConst(n, constant.UnaryOp(tokenForOp[n.Op()], nl.Val(), prec))
	}
	...
}

而 constant.UnaryOp 的代码就是根据 token.SUB (-) 计算值的:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
// UnaryOp returns the result of the unary expression op y.
// The operation must be defined for the operand.
// If prec > 0 it specifies the ^ (xor) result size in bits.
// If y is Unknown, the result is Unknown.
//
func UnaryOp(op token.Token, y Value, prec uint) Value {
	...
	case token.SUB:
		switch y := y.(type) {
		case unknownVal:
			return y
		case int64Val:
			if z := -y; z != y {
				return z // no overflow
			}
			return makeInt(newInt().Neg(big.NewInt(int64(y))))
		case intVal:
			return makeInt(newInt().Neg(y.val))
		case ratVal:
			return makeRat(newRat().Neg(y.val))
		case floatVal:
			return makeFloat(newFloat().Neg(y.val))
		case complexVal:
			re := UnaryOp(token.SUB, y.re, 0)
			im := UnaryOp(token.SUB, y.im, 0)
			return makeComplex(re, im)
		}
	...
}

终上所述,我们就大概了解其 Go iota 的递增、递减过程了。

有关 go 支持 enum 的说明和 issues

https://github.com/golang/go/issues/28987#issuecomment-497108307

参考资料

CL:

  1. https://go-review.googlesource.com/c/go/+/273008
  2. https://go-review.googlesource.com/c/go/+/275033

茶歇驿站

一个可以让你停下来看一看,在茶歇之余给你帮助的小站,这里的内容主要是后端技术,个人管理,团队管理,以及其他个人杂想。