回到简体

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,8 +1,8 @@
## 4.1. 數組
## 4.1. 数组
數組是一由固定度的特定型元素成的序列,一個數組可以由零或多元素成。因爲數組的長度是固定的因此在Go言中很少直接使用數組。和數組對應的類型是Slice切片它是可以增和收縮動態序列slice功能也更但是要理解slice工作原理的需要先理解數組
数组是一由固定度的特定型元素成的序列,一个数组可以由零或多元素成。因为数组的长度是固定的因此在Go言中很少直接使用数组。和数组对应的类型是Slice切片它是可以增和收缩动态序列slice功能也更但是要理解slice工作原理的需要先理解数组
數組的每元素可以通索引下標來訪問,索引下的范圍是從0開始到數組長度減1的位置。置的len函數將返迴數組中元素的個數
数组的每元素可以通索引下标来访问,索引下的范围是从0开始到数组长度减1的位置。置的len函数将返回数组中元素的个数
```Go
var a [3]int // array of 3 integers
@@ -20,7 +20,7 @@ for _, v := range a {
}
```
認情況下,數組的每元素都被初始化元素類型對應的零值,對於數字類型來説就是0。我也可以使用數組字面值法用一組值來初始化數組
认情况下,数组的每元素都被初始化元素类型对应的零值,对于数字类型来说就是0。我也可以使用数组字面值法用一组值来初始化数组
```Go
var q [3]int = [3]int{1, 2, 3}
@@ -28,30 +28,30 @@ var r [3]int = [3]int{1, 2}
fmt.Println(r[2]) // "0"
```
數組字面值中,如果在數組的長度位置出的是“...”省略號,則表示數組的長度是根初始化值的個數來計算。因此上面q數組的定可以簡化爲
数组字面值中,如果在数组的长度位置出的是“...”省略号,则表示数组的长度是根初始化值的个数来计算。因此上面q数组的定可以简化为
```Go
q := [...]int{1, 2, 3}
fmt.Printf("%T\n", q) // "[3]int"
```
數組的長度是數組類型的一個組成部分,因此[3]int和[4]int是兩種不同的數組類型。數組的長度必是常量表式,因爲數組的長度需要在編譯階段確定。
数组的长度是数组类型的一个组成部分,因此[3]int和[4]int是两种不同的数组类型。数组的长度必是常量表式,因为数组的长度需要在编译阶段确定。
```Go
q := [3]int{1, 2, 3}
q = [4]int{1, 2, 3, 4} // compile error: cannot assign [4]int to [3]int
```
們將會發現,數組、slice、map和結構體字面值的法都很相似。上面的形式是直接提供序初始化值序列,但是也可以指定一索引和對應值列表的方式初始化,就像下面這樣
们将会发现,数组、slice、map和结构体字面值的法都很相似。上面的形式是直接提供序初始化值序列,但是也可以指定一索引和对应值列表的方式初始化,就像下面这样
```Go
type Currency int
const (
USD Currency = iota // 美元
EUR //
GBP // 英
RMB // 人民
EUR //
GBP // 英
RMB // 人民
)
symbol := [...]string{USD: "$", EUR: "€", GBP: "£", RMB: "¥"}
@@ -59,15 +59,15 @@ symbol := [...]string{USD: "$", EUR: "€", GBP: "£", RMB: "¥"}
fmt.Println(RMB, symbol[RMB]) // "3 ¥"
```
這種形式的數組字面值形式中,初始化索引的序是無關緊要的,而且用到的索引可以省略,和前面提到的規則一樣,未指定初始值的元素用零值初始化。例如,
这种形式的数组字面值形式中,初始化索引的序是无关紧要的,而且用到的索引可以省略,和前面提到的规则一样,未指定初始值的元素用零值初始化。例如,
```Go
r := [...]int{99: -1}
```
了一含有100元素的數組r後一個元素被初始化-1其它元素都是用0初始化。
了一含有100元素的数组r后一个元素被初始化-1其它元素都是用0初始化。
如果一個數組的元素型是可以相互比的,那麽數組類型也是可以相互比的,這時候我可以直接通==比較運算符來比較兩個數組,隻有當兩個數組的所有元素都是相等的時候數組才是相等的。不相等比較運算符!=遵循同樣的規則
如果一个数组的元素型是可以相互比的,那么数组类型也是可以相互比的,这时候我可以直接通==比较运算符来比较两个数组,只有当两个数组的所有元素都是相等的时候数组才是相等的。不相等比较运算符!=遵循同样的规则
```Go
a := [2]int{1, 2}
@@ -78,7 +78,7 @@ d := [3]int{1, 2}
fmt.Println(a == d) // compile error: cannot compare [2]int == [3]int
```
爲一個眞實的例子crypto/sha256包的Sum256函數對一個任意的字slice型的數據生成一個對應的消息摘要。消息摘要有256bit大小因此對應[32]byte數組類型。如果兩個消息摘要是相同的,那可以認爲兩個消息本身也是相同(譯註:理上有HASH碰撞的情,但是實際應用可以基本忽略);如果消息摘要不同,那消息本身必然也是不同的。下面的例子用SHA256算法分别生成“x”和“X”兩個信息的摘要:
为一个真实的例子crypto/sha256包的Sum256函数对一个任意的字slice型的数据生成一个对应的消息摘要。消息摘要有256bit大小因此对应[32]byte数组类型。如果两个消息摘要是相同的,那可以认为两个消息本身也是相同(译注:理上有HASH碰撞的情,但是实际应用可以基本忽略);如果消息摘要不同,那消息本身必然也是不同的。下面的例子用SHA256算法分别生成“x”和“X”两个信息的摘要:
<u><i>gopl.io/ch4/sha256</i></u>
```Go
@@ -96,11 +96,11 @@ func main() {
}
```
上面例子中,兩個消息雖然隻有一字符的差,但是生成的消息摘要則幾乎有一半的bit位是不相同的。需要意Printf函的%x副詞參數,它用指定以十六進製的格式打印數組或slice全部的元素%t副詞參數是用打印布爾型數據,%T副詞參數是用於顯示一個值對應的數據類型。
上面例子中,两个消息虽然只有一字符的差,但是生成的消息摘要则几乎有一半的bit位是不相同的。需要意Printf函的%x副词参数,它用指定以十六进制的格式打印数组或slice全部的元素%t副词参数是用打印布尔型数据,%T副词参数是用于显示一个值对应的数据类型。
當調用一個函數的時候,函的每個調用參數將會被賦值給函數內部的參數變量,所以函數參數變量接收的是一個複製的副本,不是原始調用的量。因爲函數參數傳遞的機製導致傳遞大的數組類型將是低效的,併且對數組參數的任何的改都是生在複製的數組上,不能直接脩改調用時原始的數組變量。在這個方面Go語言對待數組的方式和其它很多編程語言不同,其它編程語言可能會隱式地將數組作爲引用或指針對象傳入被調用的函
当调用一个函数的时候,函的每个调用参数将会被赋值给函数内部的参数变量,所以函数参数变量接收的是一个复制的副本,不是原始用的量。因为函数参数传递的机制导致传递大的数组类型将是低效的,并且对数组参数的任何的改都是生在复制的数组上,不能直接修改调用时原始的数组变量。在这个方面Go语言对待数组的方式和其它很多编程语言不同,其它编程语言可能会隐式地将数组作为引用或指针对象传入被用的函
然,我可以式地入一個數組指針,那樣的話函數通過指針對數組的任何改都可以直接反饋到調用者。下面的函數用於給[32]byte型的數組清零:
然,我可以式地入一个数组指针,那样的话函数通过指针对数组的任何改都可以直接反馈到调用者。下面的函数用于给[32]byte型的数组清零:
```Go
func zero(ptr *[32]byte) {
@@ -110,7 +110,7 @@ func zero(ptr *[32]byte) {
}
```
實數組字面值[32]byte{}就可以生成一32字節的數組。而且每個數組的元素都是零值初始化也就是0。因此可以上面的zero函數寫的更簡潔一點
实数组字面值[32]byte{}就可以生成一32字节的数组。而且每个数组的元素都是零值初始化也就是0。因此可以上面的zero函数写的更简洁一点
```Go
func zero(ptr *[32]byte) {
@@ -118,8 +118,8 @@ func zero(ptr *[32]byte) {
}
```
然通過指針來傳遞數組參數是高效的,而且也允在函數內部脩改數組的值,但是數組依然是化的型,因爲數組的類型包含了化的度信息。上面的zero函數併不能接收指向[16]byte類型數組的指,而且也有任何添加或刪除數組元素的方法。由於這些原因除了像SHA256這類需要理特定大小數組的特例外,數組依然很少用作函數參數;相反,我一般使用slice替代數組
然通过指针来传递数组参数是高效的,而且也允在函数内部修改数组的值,但是数组依然是化的型,因为数组的类型包含了化的度信息。上面的zero函数并不能接收指向[16]byte类型数组的指,而且也有任何添加或删除数组元素的方法。由于这些原因除了像SHA256这类需要理特定大小数组的特例外,数组依然很少用作函数参数;相反,我一般使用slice替代数组
**練習 4.1** 編寫一個函數,計算兩個SHA256哈希中不同bit的目。(考2.6.2的PopCount函。)
**练习 4.1** 编写一个函数,计算两个SHA256哈希中不同bit的目。(考2.6.2的PopCount函。)
**練習 4.2** 編寫一個程序,默打印標準輸入的以SHA256哈希,也可以通命令行標準參數選擇SHA384或SHA512哈希算法。
**练习 4.2** 编写一个程序,默打印标准输入的以SHA256哈希,也可以通命令行标准参数选择SHA384或SHA512哈希算法。