Go应用程序的初始化是在单一的goroutine中执行的。对于包这一级别的初始化来说,在一个包里会先进行包级别变量的初始化,再进行init
函数的初始化。一个包下可以有多个init
函数,每个文件也可以有多个init
函数,多个init
函数按照它们的文件名顺序逐个初始化。但是程序不可能把所有代码都放在一个包里,通常都是会引入很多包。如果main
包引入了pkg1
包,pkg1
包本身又导入了包pkg2
,那么应用程序的初始化会按照什么顺序来执行呢?
对于这个初始化过程我粗略的画了一个示意图,理解起来更直观些 图的上半部分表示了main
包导入了pkg1
包,pkg1
包又导入了pkg2
包,这样一个包之间的依赖关系。图的下半部分表示了,这个应用初始化工作的执行时间顺序是从被导入的最深层包开始进行初始化,层层递出,最后到main
包,每个包内部的初始化程序依然是先执行包变量初始化再进行init
函数的执行
下面通过示例来验证一下这个初始化顺序,在go_tou
r目录下有三个包package1
和package2
和utils
代码目录如下:
├─package1
├─package2
└─utils
分别定义测试函数,在utils
包下有文件utils.go
go
package utils
import "fmt"
func TraceLog(t string, v int) int {
fmt.Printf("TraceLog-----%s--------%d\n", t, v)
return v
}
package1
包下有如下程序package1.go
:
go
package package1
import (
"fmt"
"go_tour/package2"
"go_tour/utils"
)
var V1 = utils.TraceLog("init package1 value1", package2.Value1+10)
var V2 = utils.TraceLog("init package1 value2", package2.Value2+10)
func init() {
fmt.Println("init func in package1")
}
package2
包下有如下程序package2.go
:
go
package package2
import (
"fmt"
"go_tour/utils"
)
var Value1 = utils.TraceLog("init package2 value1", 20)
var Value2 = utils.TraceLog("init package2 value2", 30)
func init() {
fmt.Println("init func1 in package2")
}
func init() {
fmt.Println("init func2 in package2")
}
主程序main.go
:
go
package main
import (
"fmt"
"go_tour/package1"
"go_tour/utils"
)
func init() {
fmt.Println("init func1 in main")
}
func init() {
fmt.Println("init func2 in main")
}
var MainValue1 = utils.TraceLog("init M_v1", package1.V1+10)
var MainValue2 = utils.TraceLog("init M_v2", package1.V2+10)
func main() {
fmt.Println("main func in main")
}
运行结果:
swift
TraceLog-----init package2 value1--------20
TraceLog-----init package2 value2--------30
init func1 in package2
init func2 in package2
TraceLog-----init package1 value1--------30
TraceLog-----init package1 value2--------40
init func in package1
TraceLog-----init M_v1--------40
TraceLog-----init M_v2--------50
init func1 in main
init func2 in main
main func in main
实验与结论相符合,按照导入包的层次,最先被依赖的包最先被初始化,且初始化的顺序是先初始化包变量,再说初始化init
函数。初始化过程总结如下:
- 包级别变量的初始化先于包内
init
函数的执行。 - 一个包下可以有多个
init
函数,每个文件也可以有多个init
函数。 - 多个
init
函数按照它们的文件名顺序逐个初始化。 - 初始化工作的顺序是,从被导入的最深层包开始进行初始化,层层递出最后到
main
包。 - 不管包被导入多少次,包内的
init
函数只会执行一次。 - 应用在所有初始化工作完成后才会执行
main
函数
学习交流
如果您觉得文章有帮助,点个关注哦。可以关注公众号:IT杨秀才 ,秀才后面会在公众号分享Go语言:基础 》进阶 》探秘 》实战 》面试的系列知识。也会持续更新更多硬核文章,一起聊聊互联网那些事儿!
-----------------------------历史好文-------------------------------