在Go語(yǔ)言中,可以通過import語(yǔ)句來導(dǎo)入包,導(dǎo)入的包名使用雙引號(hào)包圍,包名是從GOPATH開始計(jì)算的路徑,使用“/”進(jìn)行路徑分隔。使用import導(dǎo)入包有兩種基本格式:1、單行導(dǎo)入語(yǔ)法“import "包1" import "包2"”;2、多行導(dǎo)入語(yǔ)法“import("包1" "包2" …)”。
本教程操作環(huán)境:windows7系統(tǒng)、GO 1.18版本、Dell G3電腦。
Go語(yǔ)言import導(dǎo)入包——在代碼中使用其他的代碼
可以在一個(gè) Go語(yǔ)言源文件包聲明語(yǔ)句之后,其它非導(dǎo)入聲明語(yǔ)句之前,包含零到多個(gè)導(dǎo)入包聲明語(yǔ)句。每個(gè)導(dǎo)入聲明可以單獨(dú)指定一個(gè)導(dǎo)入路徑,也可以通過圓括號(hào)同時(shí)導(dǎo)入多個(gè)導(dǎo)入路徑。要引用其他包的標(biāo)識(shí)符,可以使用 import 關(guān)鍵字,導(dǎo)入的包名使用雙引號(hào)包圍,包名是從 GOPATH 開始計(jì)算的路徑,使用/進(jìn)行路徑分隔。
默認(rèn)導(dǎo)入的寫法
導(dǎo)入有兩種基本格式,即單行導(dǎo)入和多行導(dǎo)入,兩種導(dǎo)入方法的導(dǎo)入代碼效果是一致的。
1) 單行導(dǎo)入
單行導(dǎo)入格式如下:
import "包1" import "包2"
2) 多行導(dǎo)入
當(dāng)多行導(dǎo)入時(shí),包名在 import 中的順序不影響導(dǎo)入效果,格式如下:
import( "包1" "包2" … )
導(dǎo)入包后自定義引用的包名
如果我們想同時(shí)導(dǎo)入兩個(gè)有著名字相同的包,例如 math/rand 包和 crypto/rand 包,那么導(dǎo)入聲明必須至少為一個(gè)同名包指定一個(gè)新的包名以避免沖突。這叫做導(dǎo)入包的重命名。
import ( "crypto/rand" mrand "math/rand" // 將名稱替換為mrand避免沖突 )
導(dǎo)入包的重命名只影響當(dāng)前的源文件。其它的源文件如果導(dǎo)入了相同的包,可以用導(dǎo)入包原本默認(rèn)的名字或重命名為另一個(gè)完全不同的名字。
導(dǎo)入包重命名是一個(gè)有用的特性,它不僅僅只是為了解決名字沖突。如果導(dǎo)入的一個(gè)包名很笨重,特別是在一些自動(dòng)生成的代碼中,這時(shí)候用一個(gè)簡(jiǎn)短名稱會(huì)更方便。選擇用簡(jiǎn)短名稱重命名導(dǎo)入包時(shí)候最好統(tǒng)一,以避免包名混亂。選擇另一個(gè)包名稱還可以幫助避免和本地普通變量名產(chǎn)生沖突。例如,如果文件中已經(jīng)有了一個(gè)名為 path 的變量,那么我們可以將"path"標(biāo)準(zhǔn)包重命名為 pathpkg。
每個(gè)導(dǎo)入聲明語(yǔ)句都明確指定了當(dāng)前包和被導(dǎo)入包之間的依賴關(guān)系。如果遇到包循環(huán)導(dǎo)入的情況,Go語(yǔ)言的構(gòu)建工具將報(bào)告錯(cuò)誤。
匿名導(dǎo)入包——只導(dǎo)入包但不使用包內(nèi)類型和數(shù)值
如果只希望導(dǎo)入包,而不使用任何包內(nèi)的結(jié)構(gòu)和類型,也不調(diào)用包內(nèi)的任何函數(shù)時(shí),可以使用匿名導(dǎo)入包,格式如下:
import ( _ "path/to/package" )
其中,path/to/package 表示要導(dǎo)入的包名,下畫線_表示匿名導(dǎo)入包。
匿名導(dǎo)入的包與其他方式導(dǎo)入包一樣會(huì)讓導(dǎo)入包編譯到可執(zhí)行文件中,同時(shí),導(dǎo)入包也會(huì)觸發(fā) init() 函數(shù)調(diào)用。
包在程序啟動(dòng)前的初始化入口:init
在某些需求的設(shè)計(jì)上需要在程序啟動(dòng)時(shí)統(tǒng)一調(diào)用程序引用到的所有包的初始化函數(shù),如果需要通過開發(fā)者手動(dòng)調(diào)用這些初始化函數(shù),那么這個(gè)過程可能會(huì)發(fā)生錯(cuò)誤或者遺漏。我們希望在被引用的包內(nèi)部,由包的編寫者獲得代碼啟動(dòng)的通知,在程序啟動(dòng)時(shí)做一些自己包內(nèi)代碼的初始化工作。
例如,為了提高數(shù)學(xué)庫(kù)計(jì)算三角函數(shù)的執(zhí)行效率,可以在程序啟動(dòng)時(shí),將三角函數(shù)的值提前在內(nèi)存中建成索引表,外部程序通過查表的方式迅速獲得三角函數(shù)的值。但是三角函數(shù)索引表的初始化函數(shù)的調(diào)用不希望由每一個(gè)外部使用三角函數(shù)的開發(fā)者調(diào)用,如果在三角函數(shù)的包內(nèi)有一個(gè)機(jī)制可以告訴三角函數(shù)包程序何時(shí)啟動(dòng),那么就可以解決初始化的問題。
Go 語(yǔ)言為以上問題提供了一個(gè)非常方便的特性:init() 函數(shù)。
init() 函數(shù)的特性如下:
-
每個(gè)源碼可以使用 1 個(gè) init() 函數(shù)。
-
init() 函數(shù)會(huì)在程序執(zhí)行前(main() 函數(shù)執(zhí)行前)被自動(dòng)調(diào)用。
-
調(diào)用順序?yàn)?main() 中引用的包,以深度優(yōu)先順序初始化。
例如,假設(shè)有這樣的包引用關(guān)系:main→A→B→C,那么這些包的 init() 函數(shù)調(diào)用順序?yàn)椋?/p>
C.init→B.init→A.init→main
說明:
-
同一個(gè)包中的多個(gè) init() 函數(shù)的調(diào)用順序不可預(yù)期。
-
init() 函數(shù)不能被其他函數(shù)調(diào)用。
理解包導(dǎo)入后的init()函數(shù)初始化順序
Go 語(yǔ)言包會(huì)從 main 包開始檢查其引用的所有包,每個(gè)包也可能包含其他的包。Go 編譯器由此構(gòu)建出一個(gè)樹狀的包引用關(guān)系,再根據(jù)引用順序決定編譯順序,依次編譯這些包的代碼。
在運(yùn)行時(shí),被最后導(dǎo)入的包會(huì)最先初始化并調(diào)用 init() 函數(shù)。
通過下面的代碼理解包的初始化順序。
代碼8-3 包導(dǎo)入初始化順序入口(…/chapter08/pkginit/main.go)
package main import "chapter08/code8-2/pkg1" func main() { pkg1.ExecPkg1() }
代碼說明如下:
-
第 3 行,導(dǎo)入 pkg1 包。
-
第 7 行,調(diào)用 pkg1 包的 ExecPkg1() 函數(shù)。
代碼8-4 包導(dǎo)入初始化順序pkg1(…/chapter08/pkginit/pkg1/pkg1.go)
package pkg1 import ( "chapter08/code8-2/pkg2" "fmt" ) func ExecPkg1() { fmt.Println("ExecPkg1") pkg2.ExecPkg2() } func init() { fmt.Println("pkg1 init") }
代碼說明如下:
-
第 4 行,導(dǎo)入 pkg2 包。
-
第 8 行,聲明 ExecPkg1() 函數(shù)。
-
第 12 行,調(diào)用 pkg2 包的 ExecPkg2() 函數(shù)。
-
第 15 行,在 pkg1 包初始化時(shí),打印 pkg1 init。
代碼8-5 包導(dǎo)入初始化順序pkg2(…/chapter08/pkginit/pkg2/pkg2.go)
package pkg2 import "fmt" func ExecPkg2() { fmt.Println("ExecPkg2") } func init() { fmt.Println("pkg2 init") }
代碼說明如下:
-
第 5 行,聲明 ExecPkg2() 函數(shù)。
-
第 10 行,在 pkg2 包初始化時(shí),打印 pkg2 init。
執(zhí)行代碼,輸出如下:
pkg2 init pkg1 init ExecPkg1 ExecPkg2
【