关于golang:Go语言中的神奇函数init

27次阅读

共计 2749 个字符,预计需要花费 7 分钟才能阅读完成。

前言

哈喽,兄弟们,我是 asong。明天与大家聊一聊Go 语言中的神奇函数 init,为什么叫他神奇函数呢?因为该函数能够在所有程序执行开始前被调用,并且每个包下能够有多个init 函数。这个函数应用起来比较简单,然而你们晓得他的执行程序是怎么的嘛?本文咱们就一起来解密。

init函数的个性

先简略介绍一下 init 函数的根本个性:

  • init 函数先于 main 函数主动执行
  • 每个包中能够有多个 init 函数,每个包中的源文件中也能够有多个 init 函数
  • init函数没有输出参数、返回值,也未声明,所以无奈援用
  • 不同包的 init 函数依照包导入的依赖关系决定执行程序
  • 无论包被导入多少次,init函数只会被调用一次,也就是只执行一次

init函数的执行程序

我在刚学习 init 函数时就对他的执行程序很好奇,在谷歌上搜了几篇文章,他们都有一样的图:

下图来源于网络:

这张图片很清晰的反馈了 init 函数的加载程序:

  • 包加载优先级排在第一位,先层层递归进行包加载
  • 每个包中加载程序为:const > var > init,首先进行初始化的是常量,而后是变量,最初才是 init 函数。针对包级别的变量初始化程序,Go官网文档给出这样一个例子:
var (
    a = c + b  // == 9
    b = f()    // == 4
    c = f()    // == 5
    d = 3      // == 5 after initialization has finished
)

func f() int {
    d++
    return d
}

变量的初始化按呈现的程序从前往后进行,假若某个变量须要依赖其余变量,则被依赖的变量先初始化。所以这个例子中,初始化程序是 d -> b -> c -> a

上图只是表白了 init 函数大略的加载程序,有些细节咱们还是不晓得的,比方:以后包下有多个 init 函数,依照什么程序执行,以后源文件下有多个 init 函数,这又依照什么程序执行呢?原本想写个例子挨个验证一下的,起初一看 Go 官网文档中都有阐明,也就没有必要再写一个例子啦,间接说论断吧:

  • 如果以后包下有多个 init 函数,首先依照源文件名的字典序从前往后执行。
  • 若一个文件中呈现多个 init 函数,则依照呈现程序从前往后执行。

后面说的有点乱,对 init 函数的加载程序做一个小结:

从以后包开始,如果以后包蕴含多个依赖包,则先初始化依赖包,层层递归初始化各个包,在每一个包中,依照源文件的字典序从前往后执行,每一个源文件中,优先初始化常量、变量,最初初始化 init 函数,当呈现多个 init 函数时,则依照程序从前往后顺次执行,每一个包实现加载后,递归返回,最初在初始化以后包!

init函数的应用场景

还记得我之前的这篇文章吗:go 解锁设计模式之单例模式,借用 init 函数的加载机制咱们能够实现单例模式中的饿汉模式,具体怎么实现能够参考这篇文章,这里就不在写一遍了。

init函数的应用场景还是挺多的,比方进行服务注册、进行数据库或各种中间件的初始化连贯等。Go的规范库中也有许多中央应用到了 init 函数,比方咱们常常应用的 pprof 工具,他就应用到了 init 函数,在 init 函数外面进行路由注册:

//go/1.15.7/libexec/src/cmd/trace/pprof.go
func init() {http.HandleFunc("/io", serveSVGProfile(pprofByGoroutine(computePprofIO)))
    http.HandleFunc("/block", serveSVGProfile(pprofByGoroutine(computePprofBlock)))
    http.HandleFunc("/syscall", serveSVGProfile(pprofByGoroutine(computePprofSyscall)))
    http.HandleFunc("/sched", serveSVGProfile(pprofByGoroutine(computePprofSched)))

    http.HandleFunc("/regionio", serveSVGProfile(pprofByRegion(computePprofIO)))
    http.HandleFunc("/regionblock", serveSVGProfile(pprofByRegion(computePprofBlock)))
    http.HandleFunc("/regionsyscall", serveSVGProfile(pprofByRegion(computePprofSyscall)))
    http.HandleFunc("/regionsched", serveSVGProfile(pprofByRegion(computePprofSched)))
}

这里就不扩大太多了,更多规范库中的应用办法大家能够本人去摸索一下。

在这最初总结一下应用 init 要留神的问题吧:

  • 编程时不要依赖 init 的程序
  • 一个源文件下能够有多个 init 函数,代码比拟长时能够思考分多个 init 函数
  • 简单逻辑不倡议应用 init 函数,会减少代码的复杂性,可读性也会降落
  • init 函数中也能够启动goroutine,也就是在初始化的同时启动新的goroutine,这并不会影响初始化程序
  • init函数不应该依赖任何在 main 函数里创立的变量,因为 init 函数的执行是在 main 函数之前的
  • init函数在代码中不能被显示的调用,不能被援用(赋值给函数变量),否则会呈现编译谬误。
  • 导入包不要呈现循环依赖,这样会导致程序编译失败
  • Go程序仅仅想要用一个 packageinit执行,咱们能够这样应用:import _ "test_xxxx",导入包的时候加上下划线就 ok
  • 包级别的变量初始化、init函数执行,这两个操作都是在同一个 goroutine 中调用的,按顺序调用,一次一个包

总结

好啦,这篇文章到这里就完结了,自身 init 函数就很好了解,写这篇文章的目标就是让大家理解他的执行程序,这样在日常开发中才不会写出bug。心愿本文对大家有所帮忙,咱们下期见!

素质三连(分享、点赞、在看)都是笔者继续创作更多优质内容的能源!我是asong,咱们下期见。

创立了一个 Golang 学习交换群,欢送各位大佬们踊跃入群,咱们一起学习交换。入群形式:关注公众号获取。更多学习材料请到公众号支付。

举荐往期文章:

  • Go 语言如何实现可重入锁?
  • Go 语言中 new 和 make 你应用哪个来分配内存?
  • 源码分析 panic 与 recover,看不懂你打我好了!
  • 空构造体引发的大型打脸现场
  • Leaf—Segment 分布式 ID 生成零碎(Golang 实现版本)
  • 面试官:两个 nil 比拟后果是什么?
  • 面试官:你能用 Go 写段代码判断以后零碎的存储形式吗?
  • 如何平滑切换线上 Elasticsearch 索引

正文完
 0