★文章每週持續更新,原創不易,「三連」讓更多人看到是對我最大的確定。能夠微信搜索公衆號「 後端技術學堂 」第一時間閱讀(通常比博客早更新一到兩篇)web
」
對於通常的語言使用者來講 ,20% 的語言特性就可以知足 80% 的使用需求,剩下在使用中掌握。基於這一理論,Go 基礎系列的文章不會刻意追求面面俱到,但該有知識點都會覆蓋,目的是帶你快跑遇上 Golang 這趟新車。編程
Hurry up , Let's go !後端
前面咱們學習過 Golang 中基礎數據類型,好比內置類型 int
string
bool
等,其實還有一些複雜一點點,但很好用的複合類型,相似 C 中的數組和 struct
、C++ 中的 map
,今天咱們就來學習 Go 中的複合類型。數組
經過本文的學習你將掌握如下知識:微信
指針不保存實際數據的內容,而是保存了指向值的內存地址 。用 &
對變量取內存地址,用 *
來訪問指向的內存。這點和 C 中的指針是同樣,惟一不一樣的是 Go 中的指針不能運算。app
a := 3
pa := &a // 用 `&` 對變量取內存地址
fmt.Println("point", a, *pa) // 用 `*` 來訪問指向的內存
只聲明沒賦值的指針值是 nil
,表明空指針。編輯器
var a0 *int // 只聲明沒賦值的指針是nil
if a0 == nil {
fmt.Println("point", "it is nil point")
}
與C中的結構體相似, 結構體是一種聚合的數據類型,是由零個或多個任意類型的值聚合成的實體。每一個值稱爲結構體的成員,看例子:函數
type Test struct {
a int
b int
}
語法上的不一樣看到了嗎? 每一個結構體字段以後沒有分號,沒有分號寫起來仍是很舒服的。學習
能夠在定義的時候初始化ui
test := Test{1, 2} // 定義結構體變量並初始化
初始化部分結構體字段
t2 = Test{a: 3} //指定賦值Test.a爲3 Test.b隱式賦值0
隱式初始化
t3 = Test{} // .a .b都隱式賦值0
多個變量能夠分組一塊兒賦值
var (
t1 = Test{8, 6}
t2 = Test{a: 3} //指定賦值Test.a Test.b隱式賦值0
t3 = Test{} // .a .b都隱式賦值0
pt4 = &Test{8, 6} // 指針
)
經過 .
運算來訪問結構體成員,不區分結構體類型或是結構體指針類型。
fmt.Println("struct", st0.a, st0.b) // 經過 . 運算來訪問結構體成員
對於只聲明沒賦值的結構體,其內部變量被賦予零值,下面咱們聲明瞭 st0
但沒有對其賦值。
var st0 Test
fmt.Println("struct", st0.a, st0.b) //輸出:struct 0 0
數組是一個由固定長度的特定類型元素組成的序列,一個數組能夠由零個或多個元素組成。 數組能夠用下標訪問元素,下標從 0 開始。
數組聲明後賦值
var strarr [2]string // 數組聲明語法
strarr[0] = "ready"
strarr[1] = "go"
聲明賦值同時完成
intarr := [5]int{6, 8, 9, 10, 7} // 聲明賦值同時完成
對於肯定初始值個數的數組,能夠省略數組長度
intarr := [...]int{6, 8, 9, 10, 7} // 聲明賦值同時完成
切片是變長的序列,序列中每一個元素都有相同的類型。slice
語法和數組很像,只是沒有固定長度而已,切片底層引用一個數組對象,修改切片會修改原數組。
經過切片能夠訪問數組的部分或所有元素,正由於切片長度不是固定的,所以切片比數組更加的經常使用。
簡短聲明並初始化切片
s0 := []int{1, 2, 3, 4, 5, 6} // 簡短聲明加賦值
聲明後再初始化
var s []int // 聲明切片s
s = s0 // 用切片s0初始化切片s
聲明並初始化切片
var s00 []int = s0 // 用切片s0初始化切片s
切片的零值是 nil
// 切片的零值是nil 空切片長度和容量都是0
var nilslice []int
if nilslice == nil {
fmt.Println("slice", "nilslice is nil ", len(nilslice), cap(nilslice))
}
除了上述的常規初始化方法,還能夠用 make
內置函數來建立切片
// 內建函數make建立切片,指定切片長度和容量
// make 函數會分配一個元素爲零值的數組並返回一個引用了它的切片
s2 := make([]int, 4, 6) //建立元素都是0的切片s2, 長度爲4,容量爲6 第三個參數能夠省略
fmt.Println("slice", len(s2), cap(s2), s2)
長度表示切片中元素的數目,可用內置函數 len
函數獲得。
容量表示切片中第一個元素到引用的底層數組結尾所包含元素個數,可用內置函數 cap
求得。
切片區間遵循「左閉右開」原則,
s0 := [5]int{6, 8, 9, 10, 7} // 數組定義
var slice []int = intarr[1:4] // 建立切片slice 包含數組子序列
默認上下界。切片下界的默認值爲 0,上界默認是該切片的長度。
fmt.Println("slice", s0[:], s0[0:], s0[:5], s0[0:5]) // 這四個切片相同
append 函數用於在切片末尾追加新元素。
添加元素也分兩種狀況。
s2 := make([]int, 4, 6) //建立元素都是0的切片s2, 長度爲4,容量爲6 第三個參數能夠省略
s22 := append(s2, 2) // append每次都是在最後添加,因此此時,s21 s22指向同一個底層數組
fmt.Println(s21, s22) // [0 0 0 0 2] [0 0 0 0 2]
此時會分配新的數組空間,並返回指向這個新分配的數組的切片。
下面例子中 s24 切片已經指向新分配的數組,s22 依然指向的是原來的數組空間,而 s24 已經指向了新的底層數組。
s24 := append(s2, 1, 2, 3)
fmt.Println(s24, s22) // s24 [0 0 0 0 1 2 3] [0 0 0 0 2]
能夠定義切片的切片,相似其餘語言中的二維數組用法。參考代碼:
s3 := [][]int{
{1, 1, 1},
{2, 2, 2},
}
fmt.Println(s3, s3[0], len(s3), cap(s3)) // 輸出: [[1 1 1] [2 2 2]] [1 1 1] 2 2
在 Go 中 map
是鍵值對類型,表明 key
和value
的映射關係,一個map就是一個哈希表的引用 。
下面這樣定義並初始化一個 map 變量
m0 := map[int]string{
0: "0",
1: "1",
}
也能夠用內置 make 函數來初始化一個 map 變量,後續再向其中添加鍵值對。像下面這樣:
m1 := make(map[int]string) // make 函數會返回給定類型的映射,並將其初始化備用
if m1 != nil {
fmt.Println("map", "m1 is not nil", m1) // m1 不是nil
}
m1[0] = "1"
m1[1] = "2"
注意:只聲明不初始化的map變量是 nil
映射,不能直接拿來用!
var m map[int]string // 未初始化的m零值是nil映射
if m == nil {
fmt.Println("map", "m is nil", m)
}
//m[0] = "1" // 這句引起panic異常, 映射的零值爲 nil 。nil映射既沒有鍵,也不能添加鍵。
使用語法:vaule= m[key]
獲取鍵 key 對應的元素 vaule 。
上面咱們只用了一個變量來獲取元素,其實這個操做會返回兩個值,第一個返回值表明讀書的元素,第二個返回值是表明鍵是否存在的 bool 類型,舉例說明:
v, st := m1[0] // v是元素值,下標對應的元素存在st=true 不然st=false
_, st1 := m1[0] // _ 符號表示忽略第一個元素
v1, _ := m1[0] // _ 符號表示忽略第二個元素
fmt.Println(v, st, v1, st1, m1[2]) // m1[2]不存在,返回元素string的零值「空字符」
內置函數 delete
能夠刪除 map 元素,舉例:
delete(m1, 1) // 刪除鍵是 1 的元素
range 用於遍歷 切片 或 映射。
當使用for
循環和 range
遍歷數組或切片時,每次迭代都會返回兩個值。第一個值爲當前元素的下標,第二個值爲該下標所對應元素的一份副本。
s1 := []int{1, 2, 3, 4, 5, 6}
for key, vaule := range s1 {
fmt.Println("range", key, vaule)
}
for key := range s1 { // 只須要索引,忽略第二個變量便可
fmt.Println("range", key)
}
for _, vaule := range s1 { // 只須要元素值,用'_'忽略索引
fmt.Println("range", vaule)
}
當使用for
循環和 range
遍歷map
時,每次迭代都會返回兩個值。第一個值爲當前元素 key
, 第二個值是 value
。
m0 := map[int]string{
0: "0",
1: "1",
}
fmt.Println("map", m0)
for k, v := range m0 { // range遍歷映射,返回key 和 vaule
fmt.Println("map", "m0 key:", k, "vaule:", v)
}
經過本文的學習,咱們掌握了 Golang 中基本的控制流語句,利用這些控制語句加上一節介紹的變量等基礎知識,能夠構成豐富的程序邏輯,你就能用 Golang 來作一些有意思的事情了。
感謝各位的閱讀,文章的目的是分享對知識的理解,技術類文章我都會反覆求證以求最大程度保證準確性,若文中出現明顯紕漏也歡迎指出,咱們一塊兒在探討中學習.
今天的技術分享就到這裏,咱們下期再見。
創做不易,白票不是好習慣,若是在我這有收穫,動動手指「點贊」「關注」是對我持續創做的最大支持。
★能夠微信搜索公衆號「 後端技術學堂 」回覆「資料」「1024」有我給你準備的各類編程學習資料。文章每週持續更新,咱們下期見!
」
本文使用 mdnice 排版