Go?的入口函數(shù)和包初始化的使用
包 package
- Go 包是 Go 語言的基本組成單元,一個 Go 程序就是一組包的集合,所有 Go 代碼都位于包中
- Go 源碼可以導入其他 Go 包,并使用其中的導出語法元素,包括類型、變量、函數(shù)、方法等,而且 main 函數(shù)是整個 Go 應(yīng)用的入口函數(shù)
- Go 語言提供了很多內(nèi)置包,如 fmt、os、io 等
- 任何源代碼文件必須屬于某個包,同時源碼文件的第一行有效代碼必須是
package 包名
語句,通過該語句聲明源碼文件所在的包
main.main 函數(shù):Go 應(yīng)用的入口函數(shù)
- Go 語言中有一個特殊的函數(shù):main 包中的 main 函數(shù),也就是
main.main
,它是所有 Go 可執(zhí)行程序的用戶層執(zhí)行邏輯的入口函數(shù) - Go 程序在用戶層面的執(zhí)行邏輯,會在這個函數(shù)內(nèi)按照它的調(diào)用順序展開
package main
- 整個 Go 可執(zhí)行程序中僅允許存在一個名為 main 的包
package main
想要引用別的包的代碼,必須同樣以包的方式進行引用
package main func main() { // 用戶層執(zhí)行邏輯 ... ... }
Go 語言要求:可執(zhí)行程序的 main 包必須定義 main 函數(shù),否則 Go 編譯器會報錯
注意
main 包是不可以像標準庫 fmt 包那樣被導入(Import)的
其他包也可以擁有 main 函數(shù)或方法
按照 Go 的可見性規(guī)則(小寫字母卡頭的標識符為非導出標識符),非 main包中自定義的 main 函數(shù)僅限于包內(nèi)使用
package pkg1 import "fmt" func Main () { main() } func main(){ fmt.Println("main func for pkg1") }
重點
- 一個文件夾下的所有源碼文件只能屬于同一個包,不要求同名,但還是建議包名和所在目錄同名,這樣結(jié)構(gòu)更清晰,包名中不能包含特殊符號
- 給結(jié)構(gòu)定義的方法必須放在同一個包內(nèi),可以是不同文件
- 包名為 main 的包為應(yīng)用程序的入口包,編譯不包含 main 包的源碼文件時不會得到可執(zhí)行文件。
- 一個文件夾下的所有源碼文件只能屬于同一個包,屬于同一個包的源碼文件不能放在多個文件夾下
引子
不過對于 main 包的main 函數(shù)來說,還需要明確一點,就是它雖然是用戶層邏輯的入口函數(shù),但它卻不一定是用戶層第一個被執(zhí)行的函數(shù)。這是為什么呢?這跟 Go 語言的另一個函數(shù) init 有關(guān)
init 函數(shù):Go 包的初始化函數(shù)
和 main.main 函數(shù)一樣,init 函數(shù)也是一個無參數(shù)無返回值的函數(shù)
func init() { // 包初始化邏輯 ... ... }
- Go 程序會在這個包初始化的時候,自動調(diào)用它的 init 函數(shù),所以 init 函數(shù)的執(zhí)行會發(fā)生在 main 函數(shù)之前
- 在 Go 程序中不能手工顯式地調(diào)用 init,否則會收到編譯錯誤
和 main 函數(shù)不一樣
- init 函數(shù)在一個包中可以有多個,每個 Go 源文件都可以定義多個 init 函數(shù)
init 函數(shù)的執(zhí)行順序
- 在初始化 Go 包時,Go 會按照一定的順序,逐一、順序地調(diào)用這個包的 init 函數(shù)
- 一般來說,先傳遞給 Go 編譯器的源文件中的 init 函數(shù),會先被執(zhí)行;而同一個源文件中的多個 init 函數(shù),會按聲明順序依次執(zhí)行
Go 包的初始化次序
- 從程序邏輯結(jié)構(gòu)角度來看,Go 包是程序邏輯封裝的基本單元
- 每個包都可以理解為是一個“自治”的、封裝良好的、對外部暴露有限接口的基本單元
- 一個 Go 程序就是由一組包組成的,程序的初始化就是這些包的初始化
- 每個 Go 包還會有自己的依賴包、常量、變量、init 函數(shù)(其中 main 包有 main 函數(shù))等
三步走
- 依賴包按“深度優(yōu)先”的次序進行初始化
- 每個包內(nèi)按以“常量 -> 變量 -> init 函數(shù)”的順序進行初始化
- 包內(nèi)的多個 init 函數(shù)按出現(xiàn)次序進行自動調(diào)用
init 函數(shù)的特點
- 如上圖所示,執(zhí)行順位排在包內(nèi)其他語法元素(常量、變量)的后面
- 每個 init 函數(shù)在整個 Go 程序生命周期內(nèi)僅會被執(zhí)行一次
- init 函數(shù)是順序執(zhí)行的,只有當一個 init 函數(shù)執(zhí)行完畢后,才會去執(zhí)行下一個 init 函數(shù)
init 函數(shù)的用途
重置包級變量值
init 函數(shù)就好比 Go 包真正投入使用之前唯一的“質(zhì)檢員”,負責對包內(nèi)部以及暴露到外部的包級數(shù)據(jù)(主要是包級變量)的初始狀態(tài)進行檢查
實現(xiàn)對包級變量的復雜初始化
有些包級變量需要一個比較復雜的初始化過程,有些時候,使用它的類型零值或通過簡單初始化表達式不能滿足業(yè)務(wù)邏輯要求,而 init 函數(shù)則非常適合完成此項工作,標準庫 http 包中就有這樣一個典型示例
package main import ( "os" "strings" ) var ( http2VerboseLogs bool // 初始化默認值 false http2logFrameWrites bool http2logFrameReads bool http2inTests bool ) func init() { e := os.Getenv("GODEBUG") if strings.Contains(e, "http2debug=1") { http2VerboseLogs = true // 在 init 中對 http2VerboseLogs 的值進行重置 } if strings.Contains(e, "http2debug=2") { http2logFrameWrites = true http2logFrameReads = true http2inTests = true } }
http 包在init 函數(shù)中,就根據(jù)環(huán)境變量 GODEBUG 的值,對這些包級開關(guān)變量進行了復雜的初始化,從而保證了這些開關(guān)變量在 http 包完成初始化后,可以處于合理狀態(tài)
在 init 函數(shù)中實現(xiàn)“注冊模式”
來看一段使用 lib/pq 包訪問 PostgreSQL 數(shù)據(jù)庫的代碼 ??
package main import ( "database/sql" "log" _ "github.com/lib/pq" ) func main() { db, err := sql.Open("postgres", "user=pqgotest dbname=pqgotest sslmode=ver) if err != nil { log.Fatal(err) } age := 21 rows, err := db.Query("SELECT name FROM users WHERE age = $1", age) } 復制代碼
這里是以空導入_
的方式導入 lib/pq 包的,main 函數(shù)中沒有使用 pq 包的任何變量、函數(shù)或方法,這樣就實現(xiàn)了對 PostgreSQL數(shù)據(jù)庫的訪問
實際原因
在 pq 包的 conn.go 源碼文件中的 init 函數(shù)
func init() { sql.Register("postgres", &Driver{}) }
- 利用了空導入的特性,將 lib/pq 包作為 main 包的依賴包,在包初始化時,會先執(zhí)行 lib/pq 包里面的 init 函數(shù)
- pq 包的 init 函數(shù)將自己實現(xiàn)的 sql 驅(qū)動注冊到了 sql 包中
- 這樣在實際應(yīng)用代碼中,Open 數(shù)據(jù)庫時,傳入驅(qū)動名字(這里是 postgres),就能得到數(shù)據(jù)庫實例,然后對數(shù)據(jù)庫進行操作,實際上是因為調(diào)用了 pq 包中相應(yīng)的驅(qū)動實現(xiàn)的
好處:這種通過在 init 函數(shù)中注冊自己的實現(xiàn)的模式,就有效降低了 Go 包對外的直接暴露,尤其是包級變量的暴露,從而避免了外部通過包級變量對包狀態(tài)的改動
工廠設(shè)計模式
從標準庫 database/sql 包的角度來看,這種“注冊模式”實質(zhì)是一種工廠設(shè)計模式的實現(xiàn),sql.Open
函數(shù)就是這個模式中的工廠方法,它根據(jù)外部傳入的驅(qū)動名稱“生產(chǎn)”出不同類別的數(shù)據(jù)庫實例句柄
通過注冊模式實現(xiàn)獲取各種格式圖片的寬、高
Go 源碼
package main import ( "fmt" "image" _ "image/gif" _ "image/jpeg" _ "image/png" "os" ) func main() { // 支持 png、jpeg、gif width, height, err := imageSize(os.Args[1]) if err != nil { fmt.Println("get image size error:", err) return } fmt.Printf("image size: [%d,%d]\n", width, height) } func imageSize(imageFile string) (int, int, error) { // 打開圖片文件 f, _ := os.Open(imageFile) defer f.Close() // 對文件進行解碼,得到圖片實例 img, _, err := image.Decode(f) if err != nil { return 0, 0, err } // 返回圖片區(qū)域 b := img.Bounds() return b.Max.X, b.Max.Y, nil }
- 上面的源碼支持 png、jpeg、gif 三種格式的圖片
- 但并不需要手動支持圖片格式
- 是因為 image/png、image/jpeg 和 image/gif 包都在各自的 init 函數(shù)中,將自己“注冊”到 image 的支持格式列表中了
// $GOROOT/src/image/png/reader.go func init() { image.RegisterFormat("png", pngHeader, Decode, DecodeConfig) } // $GOROOT/src/image/jpeg/reader.go func init() { image.RegisterFormat("jpeg", "\xff\xd8", Decode, DecodeConfig) } // $GOROOT/src/image/gif/reader.go func init() { image.RegisterFormat("gif", "GIF8?a", Decode, DecodeConfig) }
到此這篇關(guān)于Go 的入口函數(shù)和包初始化的使用的文章就介紹到這了,更多相關(guān)Go 入口函數(shù)和包初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
分析Go語言中CSP并發(fā)模型與Goroutine的基本使用
我們都知道并發(fā)是提升資源利用率最基礎(chǔ)的手段,尤其是當今大數(shù)據(jù)時代,流量對于一家互聯(lián)網(wǎng)企業(yè)的重要性不言而喻。串流顯然是不行的,尤其是對于web后端這種流量的直接載體。并發(fā)是一定的,問題在于怎么執(zhí)行并發(fā)。常見的并發(fā)方式有三種,分別是多進程、多線程和協(xié)程2021-06-06golang之數(shù)據(jù)校驗的實現(xiàn)代碼示例
這篇文章主要介紹了golang之數(shù)據(jù)校檢的實現(xiàn)代碼示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-10-10Go語言開發(fā)瀏覽器視頻流rtsp轉(zhuǎn)webrtc播放
這篇文章主要為大家介紹了Go語言開發(fā)瀏覽器視頻流rtsp轉(zhuǎn)webrtc播放的過程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-04-04golang中為什么Response.Body需要被關(guān)閉詳解
這篇文章主要給大家介紹了關(guān)于golang中為什么Response.Body需要被關(guān)閉的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2018-08-08