淺談golang package中init方法的多處定義及運(yùn)行順序問(wèn)題
在不了解這個(gè)問(wèn)題之前,在網(wǎng)上搜索一下竟然搜出了兩個(gè)完全相反的結(jié)果,所以打算自己測(cè)試下這個(gè)問(wèn)題。
首先給出結(jié)論:
在同一個(gè)package中,可以多個(gè)文件中定義init方法
在同一個(gè)go文件中,可以重復(fù)定義init方法
在同一個(gè)package中,不同文件中的init方法的執(zhí)行按照文件名先后執(zhí)行各個(gè)文件中的init方法
在同一個(gè)文件中的多個(gè)init方法,按照在代碼中編寫的順序依次執(zhí)行不同的init方法
下面看下測(cè)試的代碼:
在當(dāng)前目錄下新建main.go及testinit目錄,在testinit目錄下共有三個(gè)文件:123.go、ini1.go、ini2.go,各個(gè)源碼文件分別如下:
123.go
package testinit
import "fmt"
func init(){
fmt.Println("123init")
}
ini1.go
package testinit
import "fmt"
func init(){
fmt.Println("init1")
}
func init(){
fmt.Println("init1-2")
}
ini2.go
package testinit
import "fmt"
func init(){
fmt.Println("init2")
}
main.go
package main
import (
_ "./testinit"
"fmt"
)
func main(){
fmt.Println("main")
}
如上main.go中導(dǎo)入testinit package,然后go build main.go,執(zhí)行顯示如下:

從運(yùn)行的結(jié)構(gòu)就能很清晰的看到,123、ini1、ini2三個(gè)文件按照文件名執(zhí)行,對(duì)于ini1.go中的兩個(gè)ini方法按照init方法編寫的先后順序執(zhí)行,最后才執(zhí)行的main方法!
補(bǔ)充:Golang中defer、return、返回值和main、init函數(shù)的陷阱
Go語(yǔ)言中延遲函數(shù)defer充當(dāng)著 cry...catch 的重任,使用起來(lái)也非常簡(jiǎn)便,然而在實(shí)際應(yīng)用中,很多gopher并沒(méi)有真正搞明白defer、return和返回值之間的執(zhí)行順序。他們的特點(diǎn):
多個(gè)defer的執(zhí)行順序?yàn)椤昂筮M(jìn)先出”;
defer、return、返回值三者的執(zhí)行邏輯應(yīng)該是:return最先執(zhí)行,return負(fù)責(zé)將結(jié)果寫入返回值中;接著defer開(kāi)始執(zhí)行一些收尾工作;最后函數(shù)攜帶當(dāng)前返回值退出。
如何解釋兩種結(jié)果的不同:
上面兩段代碼的返回結(jié)果之所以不同,其實(shí)從上面第2條結(jié)論很好理解。
a()int 函數(shù)的返回值沒(méi)有被提前聲名,其值來(lái)自于其他變量的賦值,而defer中修改的也是其他變量,而非返回值本身,因此函數(shù)退出時(shí)返回值并沒(méi)有被改變。
b()(i int) 函數(shù)的返回值被提前聲名,也就意味著defer中是可以調(diào)用到真實(shí)返回值的,因此defer在return賦值返回值 i 之后,再一次地修改了 i 的值,最終函數(shù)退出后的返回值才會(huì)是defer修改過(guò)的值。
package main
import (
"fmt"
)
func main() {
fmt.Println("c return:", *(c())) // 打印結(jié)果為 c return: 2
}
func c() *int {
var i int
defer func() {
i++
fmt.Println("c defer2:", i) // 打印結(jié)果為 c defer: 2
}()
defer func() {
i++
fmt.Println("c defer1:", i) // 打印結(jié)果為 c defer: 1
}()
return &i
}
雖然 c()*int 的返回值沒(méi)有被提前聲明,但是由于 c()*int 的返回值是指針變量,那么在return將變量 i 的地址賦給返回值后,defer再次修改了 i 在內(nèi)存中的實(shí)際值,因此函數(shù)退出時(shí)返回值雖然依舊是原來(lái)的指針地址,但是其指向的內(nèi)存實(shí)際值已經(jīng)被成功修改了。
Go里面有兩個(gè)保留的函數(shù):init函數(shù)(能夠應(yīng)用于所有的package)和main函數(shù)(只能應(yīng)用于package main)。這兩個(gè)函數(shù)在定義時(shí)不能有任何的參數(shù)和返回值。雖然一個(gè)package里面可以寫任意多個(gè)init函數(shù),但這無(wú)論是對(duì)于可讀性還是以后的可維護(hù)性來(lái)說(shuō),我們都強(qiáng)烈建議用戶在一個(gè)package中每個(gè)文件只寫一個(gè)init函數(shù)。
Go程序會(huì)自動(dòng)調(diào)用init()和main(),所以你不需要在任何地方調(diào)用這兩個(gè)函數(shù)。每個(gè)package中的init函數(shù)都是可選的,但package main就必須包含一個(gè)main函數(shù)。
程序的初始化和執(zhí)行都起始于main包。如果main包還導(dǎo)入了其它的包,那么就會(huì)在編譯時(shí)將它們依次導(dǎo)入。有時(shí)一個(gè)包會(huì)被多個(gè)包同時(shí)導(dǎo)入,那么它只會(huì)被導(dǎo)入一次(例如很多包可能都會(huì)用到fmt包,但它只會(huì)被導(dǎo)入一次,因?yàn)闆](méi)有必要導(dǎo)入多次)。
當(dāng)一個(gè)包被導(dǎo)入時(shí),如果該包還導(dǎo)入了其它的包,那么會(huì)先將其它包導(dǎo)入進(jìn)來(lái),然后再對(duì)這些包中的包級(jí)常量和變量進(jìn)行初始化,接著執(zhí)行init函數(shù)(如果有的話),依次類推。
等所有被導(dǎo)入的包都加載完畢了,就會(huì)開(kāi)始對(duì)main包中的包級(jí)常量和變量進(jìn)行初始化,然后執(zhí)行main包中的init函數(shù)(如果存在的話),最后執(zhí)行main函數(shù)。
下圖詳細(xì)地解釋了整個(gè)執(zhí)行過(guò)程:

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
golang?select?機(jī)制和超時(shí)問(wèn)題
golang 中的協(xié)程使用非常方便,但是協(xié)程什么時(shí)候結(jié)束是一個(gè)控制問(wèn)題,可以用 select 配合使用,這篇文章主要介紹了golang?select?機(jī)制和超時(shí)問(wèn)題,需要的朋友可以參考下2022-06-06
使用Go語(yǔ)言實(shí)現(xiàn)benchmark解析器
這篇文章主要為大家詳細(xì)介紹了如何使用Go語(yǔ)言實(shí)現(xiàn)benchmark解析器并實(shí)現(xiàn)及Web UI 數(shù)據(jù)可視化,文中的示例代碼講解詳細(xì),需要的小伙伴可以參考一下2025-04-04
使用Go語(yǔ)言實(shí)現(xiàn)并發(fā)處理CSV文件到數(shù)據(jù)庫(kù)
Go?語(yǔ)言的?goroutine?和通道(channel)非常適合用來(lái)并發(fā)地處理數(shù)據(jù),本文將通過(guò)簡(jiǎn)單示例介紹一下如何使用Go語(yǔ)言并發(fā)地處理?CSV?文件并將數(shù)據(jù)插入到數(shù)據(jù)庫(kù)中,感興趣的可以了解下2025-01-01
在Golang中實(shí)現(xiàn)RSA算法的加解密操作詳解
RSA 是一種非對(duì)稱加密算法,廣泛使用于數(shù)據(jù)的安全傳輸,crypto/rsa 是 Golang 中實(shí)現(xiàn)了 RSA 算法的一個(gè)標(biāo)準(zhǔn)庫(kù),提供了生成公私鑰對(duì)、加解密數(shù)據(jù)、簽名和驗(yàn)簽等功能,本文給大家介紹了在Golang中實(shí)現(xiàn)RSA算法的加解密操作,需要的朋友可以參考下2023-12-12
Golang報(bào)“import cycle not allowed”錯(cuò)誤的2種解決方法
這篇文章主要給大家介紹了關(guān)于Golang報(bào)"import cycle not allowed"錯(cuò)誤的2種解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以們下面隨著小編來(lái)一起看看吧2018-08-08
Golang官方限流器庫(kù)實(shí)現(xiàn)限流示例詳解
這篇文章主要為大家介紹了Golang官方限流器庫(kù)使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
Golang使用Apache PLC4X連接modbus的示例代碼
Modbus是一種串行通信協(xié)議,是Modicon公司于1979年為使用可編程邏輯控制器(PLC)通信而發(fā)表,這篇文章主要介紹了Golang使用Apache PLC4X連接modbus的示例代碼,需要的朋友可以參考下2024-07-07

