Go基礎(chǔ)教程系列之import導(dǎo)入包(遠(yuǎn)程包)和變量初始化詳解
import導(dǎo)入包
搜索路徑
import用于導(dǎo)入包:
import (
"fmt"
"net/http"
"mypkg"
)
編譯器會(huì)根據(jù)上面指定的相對(duì)路徑去搜索包然后導(dǎo)入,這個(gè)相對(duì)路徑是從GOROOT或GOPATH(workspace)下的src下開始搜索的。
假如go的安裝目錄為/usr/local/go,也就是說GOROOT=/usr/local/go,而GOPATH環(huán)境變量GOPATH=~/mycode:~/mylib,那么要搜索net/http包的時(shí)候,將按照如下順序進(jìn)行搜索:
/usr/local/go/srcnet/http ~/mycode/src/net/http ~/mylib/src/net/http
以下是go install搜索不到mypkg包時(shí)的一個(gè)報(bào)錯(cuò)信息:
can't load package: package mypkg: cannot find package "mypkg" in any of:
/usr/lib/go-1.6/src/mypkg (from $GOROOT)
/golang/src/mypkg (from $GOPATH)
也就是說,go總是先從GOROOT出先搜索,再?gòu)?code>GOPATH列出的路徑順序中搜索,只要一搜索到合適的包就理解停止。當(dāng)搜索完了仍搜索不到包時(shí),將報(bào)錯(cuò)。
包導(dǎo)入后,就可以使用這個(gè)包中的屬性。使用包名.屬性的方式即可。例如,調(diào)用fmt包中的Println函數(shù)fmt.Println。
包導(dǎo)入的過程

首先從main包開始,如果main包中有import語(yǔ)句,則會(huì)導(dǎo)入這些包,如果要導(dǎo)入的這些包又有要導(dǎo)入的包,則繼續(xù)先導(dǎo)入所依賴的包。重復(fù)的包只會(huì)導(dǎo)入一次,就像很多包都要導(dǎo)入fmt包一樣,但它只會(huì)導(dǎo)入一次。
每個(gè)被導(dǎo)入的包在導(dǎo)入之后,都會(huì)先將包的可導(dǎo)出函數(shù)(大寫字母開頭)、包變量、包常量等聲明并初始化完成,然后如果這個(gè)包中定義了init()函數(shù),則自動(dòng)調(diào)用init()函數(shù)。init()函數(shù)調(diào)用完成后,才回到導(dǎo)入者所在的包。同理,這個(gè)導(dǎo)入者所在包也一樣的處理邏輯,聲明并初始化包變量、包常量等,再調(diào)用init()函數(shù)(如果有的話),依次類推,直到回到main包,main包也將初始化包常量、包變量、函數(shù),然后調(diào)用init()函數(shù),調(diào)用完init()后,調(diào)用main函數(shù),于是開始進(jìn)入主程序的執(zhí)行邏輯。
別名導(dǎo)入和特殊的導(dǎo)入方法
當(dāng)要導(dǎo)入的包重名時(shí)會(huì)如何?例如network/convert包用于轉(zhuǎn)換從網(wǎng)絡(luò)上讀取的數(shù)據(jù),file/convert包用于轉(zhuǎn)換從文件中讀取的數(shù)據(jù),如果要同時(shí)導(dǎo)入它們,當(dāng)引用的時(shí)候指定convert.FUNC(),這個(gè)convert到底是哪個(gè)包?
可以為導(dǎo)入的包添加一個(gè)名稱屬性,為包設(shè)置一個(gè)別名。例如,除了導(dǎo)入標(biāo)準(zhǔn)庫(kù)的fmt包外,自己還定義了一個(gè)mypkg/fmt包,那么可以如下導(dǎo)入:
package main
import (
"fmt"
myfmt "mypkg/fmt"
)
func main() {
fmt.Println()
myfmt.myfunc() // 使用別名進(jìn)行訪問
}
如果不想在訪問包屬性的時(shí)候加上包名,則import導(dǎo)入的時(shí)候,可以為其設(shè)置特殊的別名:點(diǎn)(.)。
import (
. "fmt"
)
func main() {
Println() // 無需包名,直接訪問Println
}
這時(shí)要訪問fmt中的屬性,必須不能使用包名fmt。
go要求import導(dǎo)入的包必須在后續(xù)中使用,否則會(huì)報(bào)錯(cuò)。如果想要避免這個(gè)錯(cuò)誤,可以在包的前面加上下劃線:
import (
"fmt"
_ "net/http"
"mypkg"
)
這樣在當(dāng)前包中就無需使用net/http包。其實(shí)這也是為包進(jìn)行命名,只不過命名為"_",而這個(gè)符號(hào)又正好表示丟棄賦值結(jié)果,使得這成為一個(gè)匿名包。
下劃線(_)在go中,下劃線出現(xiàn)的頻率非常高,它被稱為blank identifier,可以用于賦值時(shí)丟棄值,可以用于保留import時(shí)的包,還可以用于丟棄函數(shù)的返回值。詳細(xì)內(nèi)容可參見官方手冊(cè):https://golang.org/doc/effective_go.html#blank
導(dǎo)入而不使用看上去有點(diǎn)多此一舉,但并非如此。因?yàn)閷?dǎo)入匿名包僅僅表示無法再訪問其內(nèi)的屬性。但導(dǎo)入這個(gè)匿名包的時(shí)候,會(huì)進(jìn)行一些初始化操作(例如init()函數(shù)),如果這個(gè)初始化操作會(huì)影響當(dāng)前包,那么這個(gè)匿名導(dǎo)入就是有意義的。
遠(yuǎn)程包
現(xiàn)在通過分布式版本控制系統(tǒng)進(jìn)行代碼共享是一種大趨勢(shì)。go集成了從gti上獲取遠(yuǎn)程代碼的能力。
例如:
$ go get github.com/golang/example
在import語(yǔ)句中也可以使用,首先從GOPATH中搜索路徑,顯然這是一個(gè)URL路徑,于是調(diào)用go get進(jìn)行fetch,然后導(dǎo)入。
import (
"fmt"
"github.com/golang/example"
)
當(dāng)需要從git上獲取代碼的時(shí)候,將調(diào)用go get工具自動(dòng)進(jìn)行fetch、build、install。如果workspace中已經(jīng)有這個(gè)包,那么將只進(jìn)行最后的install階段,如果沒有這個(gè)包,將保存到GOPATH的第一個(gè)路徑中,并build、install。
go get是遞歸的,所以可以直接fetch整個(gè)代碼樹。
常量和變量的初始化
Go中的常量在編譯期間就會(huì)創(chuàng)建好,即使是那些定義為函數(shù)的本地常量也如此。常量只允許是數(shù)值、字符(runes)、字符串或布爾值。
由于編譯期間的限制,定義它們的表達(dá)式必須是編譯器可評(píng)估的常量表達(dá)式(constant expression)。例如,1<<3是一個(gè)常量表達(dá)式,而math.Sin(math.Pi/4)則不是常量表達(dá)式,因?yàn)樯婕傲撕瘮?shù)math.Sin()的調(diào)用過程,而函數(shù)調(diào)用是在運(yùn)行期間進(jìn)行的。
變量的初始化和常量的初始化差不多,但初始化的變量允許是"需要在執(zhí)行期間計(jì)算的一般表達(dá)式"。例如:
var (
home = os.Getenv("HOME")
user = os.Getenv("USER")
gopath = os.Getenv("GOPATH")
)
init()函數(shù)
Go中除了保留了main()函數(shù),還保留了一個(gè)init()函數(shù),這兩個(gè)函數(shù)都不能有任何參數(shù)和返回值。它們都是在特定的時(shí)候自動(dòng)調(diào)用的,無需我們手動(dòng)去執(zhí)行。
還是這張圖:

每個(gè)包中都可以定義init函數(shù),甚至可以定義多個(gè),但建議每個(gè)包只定義一個(gè)。每次導(dǎo)入包的時(shí)候,在導(dǎo)入完成后,且變量、常量等聲明并初始化完成后,將會(huì)調(diào)用這個(gè)包中的init()函數(shù)。
對(duì)于main包,如果main包也定義了init(),那么它會(huì)在main()函數(shù)之前執(zhí)行。當(dāng)main包中的init()執(zhí)行完之后,就會(huì)立即執(zhí)行main()函數(shù),然后進(jìn)入主程序。
所以,init()經(jīng)常用來初始化環(huán)境、安裝包或其他需要在程序啟動(dòng)之前先執(zhí)行的操作。如果import導(dǎo)入包的時(shí)候,發(fā)現(xiàn)前面命名為下劃線_了,一般就說明所導(dǎo)入的這個(gè)包有init()函數(shù),且導(dǎo)入的這個(gè)包除了init()函數(shù)外,沒有其它作用。
更多關(guān)于Go語(yǔ)言中 import導(dǎo)入包(遠(yuǎn)程包)和變量初始化技術(shù)文章請(qǐng)查看下面的相關(guān)鏈接
相關(guān)文章
Go語(yǔ)言中的空值(nil)與零值(zerovalue)區(qū)別詳解
在Go語(yǔ)言中,空值(nil)和零值(zero value)是兩個(gè)不同的概念,它們?cè)谡Z(yǔ)義、使用場(chǎng)景以及實(shí)際的編程實(shí)踐中有著明顯的區(qū)別,理解這兩者的差異對(duì)于編寫清晰、健壯的Go代碼至關(guān)重要,需要的朋友可以參考下2024-06-06
Golang等多種語(yǔ)言轉(zhuǎn)數(shù)組成字符串舉例詳解
今天寫代碼遇到數(shù)組轉(zhuǎn)換成字符串操作,下面這篇文章主要給大家介紹了關(guān)于Golang等多種語(yǔ)言轉(zhuǎn)數(shù)組成字符串的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05
golang提示dial?tcp?172?.217.163.49:443:?connectex:?A?con
這篇文章主要為大家介紹了golang提示dial?tcp?172?.217.163.49:443:?connectex:?A?connection?attempt?failed解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07
Go實(shí)現(xiàn)基于RSA加密算法的接口鑒權(quán)
這篇文章主要介紹了Go實(shí)現(xiàn)基于RSA加密算法的接口鑒權(quán),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-06-06
Go語(yǔ)言對(duì)字符串進(jìn)行SHA1哈希運(yùn)算的方法
這篇文章主要介紹了Go語(yǔ)言對(duì)字符串進(jìn)行SHA1哈希運(yùn)算的方法,實(shí)例分析了Go語(yǔ)言針對(duì)字符串操作的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
go語(yǔ)言中如何使用select的實(shí)現(xiàn)示例
本文主要介紹了go語(yǔ)言中如何使用select的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05

