hello world
從一個簡單的程序開始解釋,將下面的內容放進test.go文件中,路徑隨意:
package main
import (
"fmt"
)
func main() {
fmt.Println("Hello World")
}
Go通過包的方式管理程序,每個Go源代碼文件都必須聲明自己所在的包,正如上面的package main
聲明自己所在的包是main包。
每個程序都必須有一個main包,main包作爲整個程序的編譯入口包,main包中的main()函數作爲程序的執行入口。
import關鍵字用來導入其它包,導入某個包之後就能在當前文件中使用這個包中的函數,例如上面的main包導入fmt包後,可以使用fmt包中的函數Println()。
然後可以使用go的build工具編譯這個test.go文件:
$ go build test.go
編譯後,將在當前路徑下生成一個可執行二進制文件:Windows下生成的是test.exe文件,Unix下生成的是test文件。既然是可執行文件,當然可以直接執行:
$ ./test
將輸出"Hello World"。
也可以直接通過go的run工具將編譯和運行兩個步驟合二爲一:
$ go run test.go
Hello World
go run
不會生成可執行的二進制文件,它實際上是將編譯得到的文件放進一個臨時目錄,然後執行,執行完後自動清理臨時目錄。
關於包和go文件
每個go代碼文件只能且必須使用package語句聲明一個包,也就是說一個文件中不能包含多個包。
Go中有兩種類型的包,或者說有兩種類型的文件:
- 編譯後成爲可執行文件的包,也就是main包編譯後的得到的文件
- 編譯後成爲共享庫的包,只要go程序文件中聲明的不是main包,就是庫文件
注意:
在go的官方文檔中將go的二進制可執行程序稱爲命令,有時候還會將go的源代碼文件稱爲命令的源文件。可執行程序和包相反,包一般是作爲"庫"文件存在,用於導入而非用於執行
共享庫中包含一些函數,這些函數比較通用,所以放進共享庫方便函數複用。例如fmt包中的Println函數,到處都在使用這個函數,且因爲fmt包是標準庫(Standary library),無論是誰都可以去使用這個包。
有兩種類型的庫文件:標準庫和第三方的庫。標準庫是隨Go安裝的時候放在go安裝目錄下的($GOROOT/src/
),第三方庫是放在workspace下的。關於workspace後文再說。
共享庫可以被import導入(例如fmt包)。由於導入操作是在編譯期間實現的,共享庫中不應該包含任何輸出型語句。
Go中對庫文件要求比較嚴格,或者說強制性的規範。它要求庫文件中package聲明的包名必須和目錄名稱相同,且同一個目錄下只允許有一個包,但同一個目錄下可以有多個庫文件片段,只不過這些庫文件中必須都使用package聲明它的包名爲目錄名。例如:
src/mycode
|- first.go
|- second.go
|- third.go
如果這三個文件都是庫文件,則它們都必須且只能使用package mycode
聲明自己的包爲mycode。go build
的時候,會將它們合併起來。如果聲明的包名不是mycode,go build會直接忽略它。
當然,對main包就無所謂了,它不是庫文件,可以放在任何地方,對目錄名沒有要求。但如果使用go install
,則有額外的要求,見後文。
庫文件中的大小寫命名
Go通過名稱首字母的大小寫決定屬性是否允許導出:
- 首字母大寫的屬性是允許導出的屬性
- 首字母小寫的屬性不允許被導出
所以當庫文件被導入時,只有這個庫文件中以大寫字母開頭的常量、變量、函數等纔會被導出,纔可以在其他文件中使用。
例如,庫文件abc.go中:
func first() {}
func Second() {}
當導入這個包的時候,由於first()函數首字母小寫,外界無法使用它,它只能在自己的包abc.go中使用,對外界不可見。大寫字母開頭的Second()函數會被導入,所以可用。
工作空間(workspace)
速覽
- 通過環境變量
GOPATH
設置workspace的路徑 - Go編程人員一般將它們的Go代碼放在一個
workspace
下,當然,這不是必須的 - workspace包含一個或多個版本控制系統的倉庫(如git)
- 每個倉庫包含一個或多個package
- 每個package由單個目錄下的一個或多個Go源文件組成,它們都必須聲明目錄名作爲它們的包名
- package的目錄路徑決定導入包時import的路徑
Go和其它編程語言在組織項目的時候有所不同,其它語言一般每個項目都有一個單獨的workspace,且workspace一般和版本控制倉庫進行綁定。
現在設置GOPATH環境變量,假設設置爲/gocode
echo 'export GOPATH=/gocode' >>/etc/profile.d/gopath.sh
chmod +x /etc/profile.d/gopath.sh
source /etc/profile.d/gopath.sh
go env GOPATH
確定是否正確:
$ go env GOPATH
/gocode
workspace目錄結構
每個workspace都是一個目錄,這個目錄下至少包含三個目錄:
- src:該目錄用於存放Go源代碼文件(也稱爲命令的源文件)
- bin:該目錄用於存放可執行命令(即構建後可執行的二進制go程序,也稱爲命令文件)
- pkg:該目錄用於存放共享庫文件(即構建後非可執行程序的庫包,也稱爲包對象文件)
括號中給的名稱是go官方文檔中常見的別名稱呼。
所以,先創建這3個目錄
mkdir -p /gocode/{src,pkg,bin}
GOPATH和GOROOT環境變量
GOPATH環境變量指定workspace的位置,用來指示go從哪裏搜索go源文件/包,例如import時從哪個路徑搜索包並導入。GOROOT環境變量用於指定go的安裝位置。go需要導入包時,會從GOPATH和GOROOT所設置的位置處搜索包。
默認位置爲$HOME/go
(Unix)或%USERPROFILE\go%
(Windows)。可以手動設置GOPATH環境變量的路徑從而指定workspace的位置,可以指定爲多個目錄,多個目錄時使用冒號分隔目錄(Unix系統)或使用分號分隔目錄(Windows系統)。注意,絕對不能將其設置爲go的安裝目錄,即不能和GOROOT環境變量重複。
例如,windows下設置d:\gocode
目錄爲GOPATH的路徑:
setx GOPATH d:\gocode
Unix下設置$HOME/gocode
目錄爲GOPATH的路徑:
mkdir ~/gocode
export GOPATH=~/gocode
echo 'GOPATH=~/gocode' >>~/.bashrc
go env
或go env GOPATH
命令可以輸出當前有效的GOPATH路徑。
$ go env | grep GOPATH
GOPATH="/root/gocode"
$ go env GOPATH
/root/gocode
go build
先寫兩個go文件,一個是可執行go文件test.go,一個是共享庫strutils.go,將它們放在workspace的src下。
$ mkdir -p $GOPATH/src/{hello,strutils}
$ tree -C
.
├── bin
├── pkg
├── src
│ ├── hello
│ │ └── test.go
│ └── strutils
│ └── strutils.go
注意,上面故意將test.go放在名爲hello的目錄下,可以將其放在src下的任何非庫文件目錄下(例如不能放進strutils目錄下),名稱不要求。
hello/test.go的內容如下:
package main
import (
"fmt"
"strutils"
)
func main() {
fmt.Println("Hello World")
fmt.Println(strutils.ToUpperCase("hello world"))
}
strutils/strutils.go的內容如下:
package strutils
import (
"strings"
)
func ToUpperCase(s string) string{
return strings.ToUpper(s)
}
func ToLowerCase(s string) string{
return strings.ToLower(s)
}
go build
可以用於編譯,編譯時會對import導入的包進行搜索,搜索的路徑爲標準庫所在路徑$GOROOT/src
、workspace下的src目錄。它只會生成額外的可執行文件放在當前目錄下,不會生成額外的庫文件。但需要注意,生成的可執行文件名稱可能會出乎意料:
例如進入到目錄src/hello
下,對test.go的文件進行編譯,以下三種build路徑都可用成功編譯:
cd src/hello
go build # 生成的可執行文件名爲hello
go build . # 生成的可執行文件名爲hello
go build test.go # 生成的可執行文件名爲test
前兩者是等價的,當go build
以目錄的形式進行編譯,則生成的可執行文件名爲目錄名。當go build
以go代碼文件名的方式進行編譯,則生成的可執行程序名爲go源碼文件名(去掉後綴.go或增加後綴.exe)。
go install
go還有一個工具install,go install
的操作稱爲安裝,將文件安裝到合適的位置。go install
時會先進行編譯,然後將編譯後的二進制文件保存到workspace的bin目錄下,將編譯後的庫文件(稱爲包對象文件,以".a"爲後綴)放在pkg目錄下。
注意,go install
時必須先進入到$GOPATH/src
下,且只能對目錄進行操作,不能對具體的go文件操作,因爲go認爲包和目錄名相同。給go install
指定一個目錄名,就表示編譯這個包名。
例如,對src/hello下的test.go進行安裝,由於它導入了strutils包,所以會自動將strutils也安裝好:
$ cd $GOPATH/src
$ go install hello
$ tree $GOPATH
/gocode
├── bin
│ └── hello # 二進制程序文件名爲hello,而非test
├── pkg
│ └── linux_amd64
│ └── strutils.a # 庫文件
└── src
├── hello
│ └── test.go
└── strutils
└── strutils.go
還可以單獨對庫文件進行安裝:
$ rm -rf $GOPATH/bin/* $GOPATH/pkg/*
$ cd $GOPATH/src
$ go install strutils
/gocode
├── bin
├── pkg
│ └── linux_amd64
│ └── strutils.a
└── src
├── hello
│ └── test.go
└── strutils
└── strutils.go
如果省略目錄名,則表示對當前目錄下的包進行安裝:
$ cd $GOPATH/src/hello
$ go install
再次提醒,go install
前先進入到$GOPATH/src
目錄下。
由於go install可以直接安裝二進制文件到$GOPATH/bin
,所以出於方便執行這些二進制程序,可以將這個目錄放進PATH環境變量。
$ export PATH=$PATH:`go env GOPATH`/bin
構建go程序的規範建議
1.由於可以將所有go項目放在同一個$GOPATH
目錄下,爲了區分src下的項目目錄和庫文件目錄,建議將每個項目目錄設置深一點。
例如:
bin
pkg
src
|- /first/project
|- main.go
|- myliba
|- a.go
|- b.go
|- mylibb
|- c.go
|- d.go
|- /second/project
|- main.go
|- lib
|- a.go
|- b.go
2.go install時,先進入到項目目錄下。
3.庫文件的名稱(也是目錄名)要選取合理,儘量短,但卻儘量見名知意,也儘量減少名稱重複的機率。
例如util這種名稱到處都是,可以修改爲numutil、nameutil等。