回到简体

This commit is contained in:
chai2010
2016-02-15 11:06:34 +08:00
parent 9e878f9944
commit 2b37b23285
177 changed files with 2354 additions and 2354 deletions

View File

@@ -1,18 +1,18 @@
## 4.2. Slice
Slice切片代表變長的序列,序列中每元素都有相同的型。一slice型一般作[]T其中T代表slice中元素的slice的法和數組很像,隻是沒有固定度而已。
Slice切片代表变长的序列,序列中每元素都有相同的型。一slice型一般作[]T其中T代表slice中元素的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。数组这样定义
```Go
months := [...]string{1: "January", /* ... */, 12: "December"}
```
因此一月份是months[1]十二月份是months[12]。通常,數組的第一元素索引0始,但是月份一般是從1開始的,因此我們聲明數組時直接跳第0元素第0元素被自初始化空字符串。
因此一月份是months[1]十二月份是months[12]。通常,数组的第一元素索引0始,但是月份一般是从1开始的,因此我们声明数组时直接跳第0元素第0元素被自初始化空字符串。
slice的切片操作s[i:j]其中0 ≤ i≤ j≤ cap(s),用於創建一新的slice引用s的第i元素始到第j-1元素的子序列。新的slice將隻有j-i元素。如果i位置的索引被省略的話將使用0代替如果j位置的索引被省略的話將使用len(s)代替。因此months[1:13]切片操作引用全部有效的月份和months[1:]操作等months[:]切片操作是引用整個數組。讓我們分别定表示第二季度和北方夏天月份的slice有重部分:
slice的切片操作s[i:j]其中0 ≤ i≤ j≤ cap(s),用于创建一新的slice引用s的第i元素始到第j-1元素的子序列。新的slice将只有j-i元素。如果i位置的索引被省略的话将使用0代替如果j位置的索引被省略的话将使用len(s)代替。因此months[1:13]切片操作引用全部有效的月份和months[1:]操作等months[:]切片操作是引用整个数组。让我们分别定表示第二季度和北方夏天月份的slice有重部分:
![](../images/ch4-01.png)
@@ -23,7 +23,7 @@ fmt.Println(Q2) // ["April" "May" "June"]
fmt.Println(summer) // ["June" "July" "August"]
```
兩個slice都包含了六月份下面的代是一包含相同月份的測試(性能低):
两个slice都包含了六月份下面的代是一包含相同月份的测试(性能低):
```Go
for _, s := range summer {
@@ -35,7 +35,7 @@ for _, s := range summer {
}
```
如果切片操作超出cap(s)的上限將導致一panic但是超出len(s)是意味着展了slice新slice的長度會變大:
如果切片操作超出cap(s)的上限将导致一panic但是超出len(s)是意味着展了slice新slice的长度会变大:
```Go
fmt.Println(summer[:20]) // panic: out of range
@@ -44,9 +44,9 @@ endlessSummer := summer[:5] // extend a slice (within capacity)
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。
<u><i>gopl.io/ch4/rev</i></u>
```Go
@@ -58,7 +58,7 @@ func reverse(s []int) {
}
```
里我們反轉數組的應用:
里我们反转数组的应用:
```Go
a := [...]int{0, 1, 2, 3, 4, 5}
@@ -66,7 +66,7 @@ reverse(a[:])
fmt.Println(a) // "[5 4 3 2 1 0]"
```
種將slice元素循向左镟轉n個元素的方法是三次調用reverse反轉函數,第一次是反轉開頭的n元素,然是反剩下的元素,最是反轉整個slice的元素。如果是向右循環镟轉,則將第三個函數調用移到第一個調用位置就可以了。)
种将slice元素循向左旋转n个元素的方法是三次用reverse反转函数,第一次是反转开头的n元素,然是反剩下的元素,最是反转整个slice的元素。如果是向右循环旋转,则将第三个函数调用移到第一个调用位置就可以了。)
```Go
s := []int{0, 1, 2, 3, 4, 5}
@@ -77,9 +77,9 @@ reverse(s)
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
func equal(x, y []string) bool {
@@ -95,17 +95,17 @@ func equal(x, y []string) bool {
}
```
上面關於兩個slice的深度相等測試,運行的時間併不比支持==操作的數組或字符串更多,但是何slice不直接支持比較運算符呢?方面有兩個原因。第一原因,一slice的元素是接引用的,一slice甚至可以包含自身。然有很多辦法處理這種情形,但是有一個是簡單有效的。
上面关于两个slice的深度相等测试,运行的时间并不比支持==操作的数组或字符串更多,但是何slice不直接支持比较运算符呢?方面有两个原因。第一原因,一slice的元素是接引用的,一slice甚至可以包含自身。然有很多办法处理这种情形,但是有一个是简单有效的。
第二原因,因slice的元素是接引用的,一固定值的slice在不同的時間可能包含不同的元素,因爲底層數組的元素可能會被脩改。且Go言中map等哈希表之類的數據結構的key隻做簡單的淺拷貝,它要求在整個聲明週期中相等的key必須對相同的元素。對於像指或chan之的引用型,==相等測試可以判斷兩個是否是引用相同的象。一個針對slice的相等測試的==操作符可能是有一定用的,也能臨時解決map型的key問題但是slice和數組不同的相等測試行爲會讓人睏惑。因此安全的做法是直接禁止slice之的比操作。
第二原因,因slice的元素是接引用的,一固定值的slice在不同的时间可能包含不同的元素,因为底层数组的元素可能会被修改。且Go言中map等哈希表之类的数据结构的key只做简单的浅拷贝,它要求在整个声明周期中相等的key必须对相同的元素。对于像指或chan之的引用型,==相等测试可以判断两个是否是引用相同的象。一个针对slice的相等测试的==操作符可能是有一定用的,也能临时解决map型的key问题但是slice和数组不同的相等测试行为会让人困惑。因此安全的做法是直接禁止slice之的比操作。
slice唯一合法的比操作是和nil比,例如:
slice唯一合法的比操作是和nil比,例如:
```Go
if summer == nil { /* ... */ }
```
零值的slice等nil。一nil值的slice併沒有底層數組。一nil值的slice的度和容量都是0但是也有非nil值的slice的度和容量也是0的例如[]int{}或make([]int, 3)[3:]。任意型的nil值一,我可以用[]int(nil)類型轉換表達式來生成一個對應類型slice的nil值。
零值的slice等nil。一nil值的slice并没有底层数组。一nil值的slice的度和容量都是0但是也有非nil值的slice的度和容量也是0的例如[]int{}或make([]int, 3)[3:]。任意型的nil值一,我可以用[]int(nil)类型转换表达式来生成一个对应类型slice的nil值。
```Go
var s []int // len(s) == 0, s == nil
@@ -114,16 +114,16 @@ s = []int(nil) // 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。容量部分可以省略这种情况下,容量将等于长度。
```Go
make([]T, len)
make([]T, len, cap) // same as make([]T, cap)[:len]
```
在底make建了一匿名的數組變量,然後返迴一個slice有通過返迴的slice才能引用底匿名的數組變量。在第一種語句中slice是整個數組的view。在第二個語句中slice引用了底層數組的前len元素,但是容量包含整個的數組。額外的元素是留給未來的增用的。
在底make建了一匿名的数组变量,然后返回一个slice有通过返回的slice才能引用底匿名的数组变量。在第一种语句中slice是整个数组的view。在第二个语句中slice引用了底层数组的前len元素,但是容量包含整个的数组。额外的元素是留给未来的增用的。
{% include "./ch4-02-1.md" %}