Go语言是这么做程序初始化的

Go应用程序的初始化是在单一的goroutine中执行的。对于包这一级别的初始化来说,在一个包里会先进行包级别变量的初始化,再进行init函数的初始化。一个包下可以有多个init函数,每个文件也可以有多个init函数,多个init函数按照它们的文件名顺序逐个初始化。但是程序不可能把所有代码都放在一个包里,通常都是会引入很多包。如果main包引入了pkg1包,pkg1包本身又导入了包pkg2,那么应用程序的初始化会按照什么顺序来执行呢?

对于这个初始化过程我粗略的画了一个示意图,理解起来更直观些 图的上半部分表示了main包导入了pkg1包,pkg1包又导入了pkg2包,这样一个包之间的依赖关系。图的下半部分表示了,这个应用初始化工作的执行时间顺序是从被导入的最深层包开始进行初始化,层层递出,最后到main包,每个包内部的初始化程序依然是先执行包变量初始化再进行init函数的执行

下面通过示例来验证一下这个初始化顺序,在go_tour目录下有三个包package1package2utils

代码目录如下:

复制代码
├─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语言:基础 》进阶 》探秘 》实战 》面试的系列知识。也会持续更新更多硬核文章,一起聊聊互联网那些事儿!

-----------------------------历史好文-------------------------------

《2024年必备的Go语言学习路线(建议收藏🔥)》

相关推荐
SimonKing3 分钟前
Spring Boot 动态多数据源:核心思路与关键考量
java·后端·程序员
唐叔在学习5 分钟前
为了不付费,我硬生生用AI开发了一个跨平台待办应用
后端·程序员·ai编程
武子康12 分钟前
大数据-249 离线数仓 - 电商分析 Hive 数仓实战:订单拉链表到 DWS 宽表设计与加载脚本详解
大数据·后端·apache hive
IT_陈寒30 分钟前
用Python爬虫抓了100万条数据后,我总结了这5个反封禁技巧
前端·人工智能·后端
bug攻城狮36 分钟前
SpringBoot 脚手架搭建指南:从零构建企业级开发框架
java·spring boot·后端·架构·系统架构·设计规范
人道领域41 分钟前
【苍穹外卖】深度解析:商品浏览四大核心接口设计(附完整数据流转图)
java·数据库·后端·sql
程序员爱钓鱼42 分钟前
Go静态资源嵌入方案: embed包深度解析
后端·面试·go
AskHarries44 分钟前
独立开发者最浪费时间的10件事
后端·ai编程
猹叉叉(学习版)1 小时前
【ASP.NET CORE】 12. DDD基本概念
笔记·后端·架构·c#·asp.net·.netcore
独断万古他化1 小时前
【抽奖系统开发实战】Spring Boot 项目的奖品模块开发:文件上传、时序设计与奖品创建
java·spring boot·后端·mvc·文件