美文网首页
Golang变量的初始化依赖

Golang变量的初始化依赖

作者: hezhangjian | 来源:发表于2021-06-23 18:55 被阅读0次

参考资料

翻译自

https://medium.com/golangspec/initialization-dependencies-in-go-51ae7b53f24c

正文

让我们从两个简单地go程序开始举例

程序1

package main
import "fmt"
var (
    a int = b + 1
    b int = 1
)
func main() {
    fmt.Println(a)
    fmt.Println(b)
}

程序1输出

2
1

Program exited.

程序2

package main
import "fmt"
func main() {
    var (
        a int = b + 1
        b int = 1
    )
    fmt.Println(a)
    fmt.Println(b)
}

程序2输出

./prog.go:5:17: undefined: b

Go build failed.

如果两个程序输出一样,就不会成为本文的材料了。一般情况下来说,go的初始化表达式都是从上到下,自左向右执行的。如下的例子程序

package main

import "fmt"

func f() int { fmt.Println("f"); return 1 }
func g() int { fmt.Println("g"); return 2 }
func h() int { fmt.Println("h"); return 3 }
func main() {
    var (
        a int = f()
        b int = g()
        c int = h()
    )
    fmt.Println(a, b, c)
}

输出

f
g
h
1 2 3

“一般”指的是函数内的初始化流程。如果是顶级声明(包级别),这个时候初始化依赖就要发挥作用了: 如下的程序

package main

import "fmt"

var (
    a = c - 2
    b = 2
    c = f()
)

func f() int {
    fmt.Printf("inside f and b = %d\n", b)
    return b + 1
}
func main() {
    fmt.Println(a)
    fmt.Println(b)
    fmt.Println(c)
}

并不会提示c没有被定义,而是,程序会正常运行,并按以下的流程来推断初始化顺序

  • b最先,因为它不依赖其他的未初始化变量
  • 接下来是c,c通过f函数依赖了b
  • a依赖c,所以a在最后初始化

输出是这样的

inside f and b = 2
1
2
3

每个初始化周期选择第一个(按声明顺序)准备好的变量。整个流程一直会持续,直到所有的变量都初始化完毕,或者碰到初始化循环处理不了为止,例如:

package main

import "fmt"

var (
    a = c - 2
    b = 2
    c = f()
)

func f() int {
    fmt.Printf("inside f and b = %d\n", b)
    return b + 1
}
func main() {
    fmt.Println(a)
    fmt.Println(b)
    fmt.Println(c)
}

会报错,./main.go:4:2: initialization loop:

依赖关系可以在包级别生效,即使使用多个文件也是一样的效果。

译者注:依赖关系也可以在包与包之间生效,golang不允许包与包之间的循环依赖,所以只能形成一个包内的变量依赖另一个包初始化的关系

相关文章

网友评论

      本文标题:Golang变量的初始化依赖

      本文链接:https://www.haomeiwen.com/subject/akzbyltx.html