make loop

This commit is contained in:
chai2010 2016-01-06 20:30:13 +08:00
parent 03edd41bdb
commit 336e74c44f
3 changed files with 9 additions and 9 deletions

View File

@ -1,6 +1,6 @@
## 4.1. 數組 ## 4.1. 數組
數組是一個由固定長度的特定類型元素組成的序列一個數組可以由零個或多個元素組成。因爲數組的長度是固定的因此在Go語言中很少直接使用數組。和数组对应的类型是Slice切片它是可以增長和收縮動態序列slice功能也更靈活但是要理解slice工作原理的話需要先理解數組。 數組是一個由固定長度的特定類型元素組成的序列一個數組可以由零個或多個元素組成。因爲數組的長度是固定的因此在Go語言中很少直接使用數組。和數組對應的類型是Slice切片它是可以增長和收縮動態序列slice功能也更靈活但是要理解slice工作原理的話需要先理解數組。
數組的每個元素可以通過索引下標來訪問索引下標的范圍是從0開始到數組長度減1的位置。內置的len函數將返迴數組中元素的個數。 數組的每個元素可以通過索引下標來訪問索引下標的范圍是從0開始到數組長度減1的位置。內置的len函數將返迴數組中元素的個數。
@ -97,9 +97,9 @@ func main() {
} }
``` ```
上面例子中兩個消息雖然隻有一個字符的差異但是生成的消息摘要則幾乎有一半的bit位是不相同的。需要註意Printf函數的%x副词參數它用於指定以十六進製的格式打印數組或slice全部的元素%t副词參數是用於打印布爾型數據%T副词參數是用於顯示一個值對應的數據類型。 上面例子中兩個消息雖然隻有一個字符的差異但是生成的消息摘要則幾乎有一半的bit位是不相同的。需要註意Printf函數的%x副詞參數它用於指定以十六進製的格式打印數組或slice全部的元素%t副詞參數是用於打印布爾型數據%T副詞參數是用於顯示一個值對應的數據類型。
當調用一個函數的時候,函數的每個調用參數將會被賦值給函數內部的參數變量,所以函數參數變量接收的是一個複製的副本,併不是原始調用的量。因爲函數參數傳遞的機製導致傳遞大的數組類型將是低效的併且對數組參數的任何的脩改都是發生在複製的數組上併不能直接脩改調用時原始的數組變量。在這個方面Go語言對待數組的方式和其它很多編程語言不同其它編程語言可能會隱式地將數組作爲引用或指針對象傳入被調用的函數。 當調用一個函數的時候,函數的每個調用參數將會被賦值給函數內部的參數變量,所以函數參數變量接收的是一個複製的副本,併不是原始調用的量。因爲函數參數傳遞的機製導致傳遞大的數組類型將是低效的併且對數組參數的任何的脩改都是發生在複製的數組上併不能直接脩改調用時原始的數組變量。在這個方面Go語言對待數組的方式和其它很多編程語言不同其它編程語言可能會隱式地將數組作爲引用或指針對象傳入被調用的函數。
當然,我們可以顯式地傳入一個數組指針,那樣的話函數通過指針對數組的任何脩改都可以直接反饋到調用者。下面的函數用於給[32]byte類型的數組清零 當然,我們可以顯式地傳入一個數組指針,那樣的話函數通過指針對數組的任何脩改都可以直接反饋到調用者。下面的函數用於給[32]byte類型的數組清零

View File

@ -2,7 +2,7 @@
Slice切片代表變長的序列序列中每個元素都有相同的類型。一個slice類型一般寫作[]T其中T代表slice中元素的類型語法和數組很像隻是沒有固定長度而已。 Slice切片代表變長的序列序列中每個元素都有相同的類型。一個slice類型一般寫作[]T其中T代表slice中元素的類型語法和數組很像隻是沒有固定長度而已。
數組和slice之間有着緊密的聯繫。一個slice是一個輕量級的數據结构提供了訪問數組子序列或者全部元素的功能因爲slice的底層確實引用一個數組對象。一個slice由三個部分構成指針、長度和容量。指針指向第一個slice元素對應的底層數組元素的地址意的是slice的第一個元素併不一定就是數組的第一個元素。長度對應slice中元素的數目長度不能超過容量容量一般是從slice的開始位置到底層數據的結尾位置。內置的len和cap函數分别返迴slice的長度和容量。 數組和slice之間有着緊密的聯繫。一個slice是一個輕量級的數據結構提供了訪問數組子序列或者全部元素的功能因爲slice的底層確實引用一個數組對象。一個slice由三個部分構成指針、長度和容量。指針指向第一個slice元素對應的底層數組元素的地址意的是slice的第一個元素併不一定就是數組的第一個元素。長度對應slice中元素的數目長度不能超過容量容量一般是從slice的開始位置到底層數據的結尾位置。內置的len和cap函數分别返迴slice的長度和容量。
多個slice之間可以共享底層的數據併且引用的數組部分區間可能重疊。圖4.1顯示了表示一年中每個月份名字的字符串數組還有重疊引用了該數組的兩個slice。數組這樣定義 多個slice之間可以共享底層的數據併且引用的數組部分區間可能重疊。圖4.1顯示了表示一年中每個月份名字的字符串數組還有重疊引用了該數組的兩個slice。數組這樣定義
@ -35,7 +35,7 @@ for _, s := range summer {
} }
``` ```
如果切片操作超出cap(s)的上限將導致一個panic異常但是超出len(s)則是意味着擴展了slice新slice的長度會變大 如果切片操作超出cap(s)的上限將導致一個panic異常但是超出len(s)則是意味着擴展了slice新slice的長度會變大
```Go ```Go
fmt.Println(summer[:20]) // panic: out of range fmt.Println(summer[:20]) // panic: out of range
@ -46,7 +46,7 @@ fmt.Println(endlessSummer) // "[June July August September October]"
另外,字符串的切片操作和[]byte字節類型切片的切片操作是類似的。它們都寫作x[m:n]併且都是返迴一個原始字節繫列的子序列底層都是共享之前的底層數組因此切片操作對應常量時間複雜度。x[m:n]切片操作對於字符串則生成一個新字符串如果x是[]byte的話則生成一個新的[]byte。 另外,字符串的切片操作和[]byte字節類型切片的切片操作是類似的。它們都寫作x[m:n]併且都是返迴一個原始字節繫列的子序列底層都是共享之前的底層數組因此切片操作對應常量時間複雜度。x[m:n]切片操作對於字符串則生成一個新字符串如果x是[]byte的話則生成一個新的[]byte。
因爲slice值包含指向第一個slice元素的指針因此向函數傳遞slice將允许在函數內部脩改底層數組的元素。換句話説,复制一個slice隻是對底層的數組創建了一個新的slice别名§2.3.2。下面的reverse函數在原內存空間將[]int類型的slice反轉而且它可以用於任意長度的slice。 因爲slice值包含指向第一個slice元素的指針因此向函數傳遞slice將允許在函數內部脩改底層數組的元素。換句話説,複製一個slice隻是對底層的數組創建了一個新的slice别名§2.3.2。下面的reverse函數在原內存空間將[]int類型的slice反轉而且它可以用於任意長度的slice。
```Go ```Go
gopl.io/ch4/rev gopl.io/ch4/rev
@ -80,7 +80,7 @@ fmt.Println(s) // "[2 3 4 5 0 1]"
要註意的是slice類型的變量s和數組類型的變量a的初始化語法的差異。slice和數組的字面值語法很類似它們都是用花括弧包含一繫列的初始化元素但是對於slice併沒有指明序列的長度。這會隱式地創建一個合適大小的數組然後slice的指針指向底層的數組。就像數組字面值一樣slice的字面值也可以按順序指定初始化值序列或者是通過索引和元素值指定或者的兩種風格的混合語法初始化。 要註意的是slice類型的變量s和數組類型的變量a的初始化語法的差異。slice和數組的字面值語法很類似它們都是用花括弧包含一繫列的初始化元素但是對於slice併沒有指明序列的長度。這會隱式地創建一個合適大小的數組然後slice的指針指向底層的數組。就像數組字面值一樣slice的字面值也可以按順序指定初始化值序列或者是通過索引和元素值指定或者的兩種風格的混合語法初始化。
和數組不同的是slice之不能比較,因此我們不能使用==操作符來判斷兩個slice是否含有全部相等元素。不過標準庫提供了高度優化的bytes.Equal函數來判斷兩個字節型slice是否相等[]byte但是對於其他類型的slice我們必鬚自己展開每個元素進行比較 和數組不同的是slice之不能比較,因此我們不能使用==操作符來判斷兩個slice是否含有全部相等元素。不過標準庫提供了高度優化的bytes.Equal函數來判斷兩個字節型slice是否相等[]byte但是對於其他類型的slice我們必鬚自己展開每個元素進行比較
```Go ```Go
func equal(x, y []string) bool { func equal(x, y []string) bool {
@ -115,7 +115,7 @@ s = []int(nil) // len(s) == 0, s == nil
s = []int{} // len(s) == 0, s != nil s = []int{} // len(s) == 0, s != nil
``` ```
如果你需要測試一個slice是否是空的使用len(s) == 0來判斷而不应该用s == nil來判斷。除了和nil相等比較外一個nil值的slice的行爲和其它任意0産長度的slice一樣例如reverse(nil)也是安全的。除了文檔已經明確説明的地方所有的Go語言函數應該以相同的方式對待nil值的slice和0長度的slice。 如果你需要測試一個slice是否是空的使用len(s) == 0來判斷而不應該用s == nil來判斷。除了和nil相等比較外一個nil值的slice的行爲和其它任意0産長度的slice一樣例如reverse(nil)也是安全的。除了文檔已經明確説明的地方所有的Go語言函數應該以相同的方式對待nil值的slice和0長度的slice。
內置的make函數創建一個指定元素類型、長度和容量的slice。容量部分可以省略在這種情況下容量將等於長度。 內置的make函數創建一個指定元素類型、長度和容量的slice。容量部分可以省略在這種情況下容量將等於長度。

View File

@ -1,6 +1,6 @@
# 第四章 複合數據類型 # 第四章 複合數據類型
在第三章我們討論了基本數據類型它們可以用於構建程序中數據結構是Go語言的世界的原子。在本章我們將討論複合數據類型它是以不同的方式組合基本類型可以構造出的複合數據類型。我們主要討論四種類型——數組、slice、map和結構體——同時在本章的最後我們將演示如何使用結構體來解碼和編碼到對應JSON格式的數據併且通過結合使用模闆來生成HTML頁面。 在第三章我們討論了基本數據類型它們可以用於構建程序中數據結構是Go語言的世界的原子。在本章我們將討論複合數據類型它是以不同的方式組合基本類型可以構造出的複合數據類型。我們主要討論四種類型——數組、slice、map和結構體——同時在本章的最後我們將演示如何使用結構體來解碼和編碼到對應JSON格式的數據併且通過結合使用模闆來生成HTML頁面。
數組和結構體是聚合類型它們的值由許多元素或成員字段的值組成。數組是由同構的元素組成——每個數組元素都是完全相同的類型——結構體則是由異構的元素組成的。數組和結構體都是有固定內存大小的數據結構。相比之下slice和map則是動態的數據結構它們將根據需要動態增長。 數組和結構體是聚合類型它們的值由許多元素或成員字段的值組成。數組是由同構的元素組成——每個數組元素都是完全相同的類型——結構體則是由異構的元素組成的。數組和結構體都是有固定內存大小的數據結構。相比之下slice和map則是動態的數據結構它們將根據需要動態增長。