Go 中神奇的 init 函数
发布日期:2022-02-01 16:54:00 浏览次数:21 分类:技术文章

本文共 2599 字,大约阅读时间需要 8 分钟。

前言

哈喽,兄弟们,我是asong。今天与大家聊一聊Go语言中的神奇函数init,为什么叫他神奇函数呢?因为该函数可以在所有程序执行开始前被调用,并且每个包下可以有多个init函数。这个函数使用起来比较简单,但是你们知道他的执行顺序是怎样的嘛?本文我们就一起来解密。

init函数的特性

先简单介绍一下init函数的基本特性:

  • init函数先于main函数自动执行

  • 每个包中可以有多个init函数,每个包中的源文件中也可以有多个init函数

  • init函数没有输入参数、返回值,也未声明,所以无法引用

  • 不同包的init函数按照包导入的依赖关系决定执行顺序

  • 无论包被导入多少次,init函数只会被调用一次,也就是只执行一次

init函数的执行顺序

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

下图来源于网络:

2d39fdd7741947d767b717304eef038d.png
截屏2021-06-05 上午9.55.15

这张图片很清晰的反应了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函数的使用场景

还记得我之前的这篇文章吗:init函数的加载机制我们可以实现单例模式中的饿汉模式,具体怎么实现可以参考这篇文章,这里就不在写一遍了。

init函数的使用场景还是挺多的,比如进行服务注册、进行数据库或各种中间件的初始化连接等。Go的标准库中也有许多地方使用到了init函数,比如我们经常使用的pprof工具,他就使用到了init函数,在init函数里面进行路由注册:

//go/1.15.7/libexec/src/cmd/trace/pprof.gofunc 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,吸取他的知识 👆

转载地址:https://blog.csdn.net/slphahaha/article/details/120793438 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:Go 流水线编程模式
下一篇:什么是大端序和小端序,为什么要有字节序

发表评论

最新留言

能坚持,总会有不一样的收获!
[***.219.124.196]2024年04月18日 08时45分10秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章

C++面经总结之《Effective C++》(二) 2019-04-27
这是什么“虎狼之词”啊!!!程序员的健康问题,看一线老中医怎么说!!! 2019-04-27
打开我的收藏夹 -- Python数据分析杂谈 2019-04-27
上手Pandas,带你玩转数据(1)-- 实例详解pandas数据结构 2019-04-27
上手Pandas,带你玩转数据(2)-- 使用pandas从多种文件中读取数据 2019-04-27
上手Pandas,带你玩转数据(3)-- pandas数据存入文件 2019-04-27
爬虫遇上不让右击、不让F12的网站,该怎么办? 2019-04-27
上手Pandas,带你玩转数据(4)-- 数据清洗 2019-04-27
上手Pandas,带你玩转数据(5)-- 数据转换与数据定位 2019-04-27
上手Pandas,带你玩转数据(6)-- 摆脱对pandas可视化丑图的刻板印象吧 2019-04-27
从零开始,学会Python爬虫不再难!!! -- (1)开篇:初识爬虫,基础铺垫 丨蓄力计划 2019-04-27
从零开始,学会Python爬虫不再难!!! -- (2)承接:解析网页,抓取标签 丨蓄力计划 2019-04-27
AttributeError: module ‘urllib‘ has no attribute ‘quote‘的解决办法 2019-04-27
linux shell — 6.初识 EXT2 文件系统 2019-04-27
Java — String(字符串) 2019-04-27
linux shell — 7.linux 磁盘与文件系统管理 2019-04-27
linux shell — 8.linux 磁盘与文件系统管理(2) 2019-04-27
Java — 事件监听、事件处理 初体验 2019-04-27
linux — Centos 7(第一天) 使用时出现的问题及解决方法 2019-04-27
数据结构 — 图的概述 2019-04-27