前言
相信用過(guò)Range的朋友們都知道,Go語(yǔ)言中的range關(guān)鍵字使用起來(lái)非常的方便,它允許你遍歷某個(gè)slice或者map,并通過(guò)兩個(gè)參數(shù)(index
和value
),分別獲取到slice或者map中某個(gè)元素所在的index
以及其值。
比如像這樣的用法:
1
2
3
4
|
for index, value := range mySlice { fmt.Println( "index: " + index) fmt.Println( "value: " + value) } |
上面的例子足夠清晰的描述了range的用法,實(shí)際上在使用range關(guān)鍵字的時(shí)候,還有一些需要特別注意的地方,有一些新手很容易入的”坑”。
為了說(shuō)明這些”坑”,我們可以從下面這個(gè)稍復(fù)雜的例子說(shuō)起:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
type Foo struct { bar string } func main() { list := []Foo{ { "A" }, { "B" }, { "C" }, } list2 := make([]*Foo, len(list)) for i, value := range list { list2[i] = &value } fmt.Println(list[0], list[1], list[2]) fmt.Println(list2[0], list2[1], list2[2]) } |
在這個(gè)例子中,我們干了下面的一些事情:
1、定義了一個(gè)叫做Foo的結(jié)構(gòu),里面有一個(gè)叫bar的field。隨后,我們創(chuàng)建了一個(gè)基于Foo結(jié)構(gòu)體的slice,名字叫l(wèi)ist
2、我們還創(chuàng)建了一個(gè)基于Foo結(jié)構(gòu)體指針類(lèi)型的slice,叫做list2
3、在一個(gè)for
循環(huán)中,我們?cè)噲D遍歷list中的每一個(gè)元素,獲取其指針地址,并賦值到list2中index與之對(duì)應(yīng)的位置。
4、最后,分別輸出list與list2中的每個(gè)元素
從代碼來(lái)看,理所當(dāng)然,我們期望得到的結(jié)果應(yīng)該是這樣:
1
2
|
{A} {B} {C} &{A} &{B} &{C} |
但是結(jié)果卻出乎意料,程序的輸出是這樣的:
1
2
|
{A} {B} {C} &{C} &{C} &{C} |
從結(jié)果來(lái)看,仿佛list2中的三個(gè)元素,都指向了list中的最后一個(gè)元素。這是為什么呢?問(wèn)題就出在上面那一段for…range
循環(huán)中。
在Go的for…range
循環(huán)中,Go始終使用值拷貝的方式代替被遍歷的元素本身,簡(jiǎn)單來(lái)說(shuō),就是for…range
中那個(gè)value
,是一個(gè)值拷貝,而不是元素本身。這樣一來(lái),當(dāng)我們期望用&獲取元素的指針地址時(shí),實(shí)際上只是取到了value
這個(gè)臨時(shí)變量的指針地址,而非list
中真正被遍歷到的某個(gè)元素的指針地址。而在整個(gè)for…range
循環(huán)中,value
這個(gè)臨時(shí)變量會(huì)被重復(fù)使用,所以,在上面的例子中,list2被填充了三個(gè)相同的指針地址,并且這三個(gè)地址都指向value
,而在最后一次循環(huán)中,value
被賦與了{c}
的指針地址。因此,list2輸出的時(shí)候顯示出了三個(gè)&{c}
。
同樣的,下面的寫(xiě)法,跟for…range
的例子如出一轍:
1
2
3
4
5
|
var value Foo for var i := 0; i < len(list); i++ { value = list[i] list2[i] = &value } |
如果我們輸出list2的三個(gè)元素,結(jié)果同樣是: &{C} &{C} &{C}
那么,怎樣才是正確的寫(xiě)法呢?我們應(yīng)該用index
來(lái)訪問(wèn)for…range
中真實(shí)的元素,并獲取其指針地址:
1
2
3
|
for i, _ := range list { list2[i] = &list[i] } |
這樣,輸出list2中的元素,就能得到我們想要的結(jié)果(&{A} &{B} &{C})
了。
實(shí)驗(yàn)代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
package main import "fmt" type Foo struct { bar string } func main() { list := []Foo{ { "A" }, { "B" }, { "C" }, } list2 := make([]*Foo, len(list)) //錯(cuò)誤的例子 for i, value := range list { list2[i] = &value } //正確的例子 //for i, _ := range list { // list2[i] = &list[i] //} fmt.Println(list[0], list[1], list[2]) fmt.Println(list2[0], list2[1], list2[2]) } |
了解了range的正確使用姿勢(shì),那么我們下面這個(gè)例子也能迎刃而解了:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package main import "fmt" type MyType struct { field string } func main() { var array [10]MyType for _, e := range array { e.field = "foo" } for _, e := range array { fmt.Println(e.field) fmt.Println( "--" ) } } |
平常寫(xiě)代碼最常見(jiàn)的場(chǎng)景,就是我們需要在一個(gè)循環(huán)中修改被遍歷元素的值。比如上面這個(gè)例子,我們希望能使用for…range
循環(huán),一次性將array中每個(gè)元素的field設(shè)置為”foo”。同樣,因?yàn)閞ange值拷貝的緣故,上面的程序什么都不會(huì)輸出……
而正確的做法是:
1
2
3
|
for i, _ := range array { array[i].field = "foo" } |
通過(guò)index訪問(wèn)每個(gè)元素,并修改其field,這樣,就能輸出一堆”foo”了……
實(shí)驗(yàn)代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
package main import "fmt" type MyType struct { field string } func main() { var array [10]MyType for i, _ := range array { array[i].field = "foo" } for _, e := range array { fmt.Println(e.field) } } |
總結(jié)
以上就是關(guān)于Go語(yǔ)言中Range關(guān)鍵字的全部?jī)?nèi)容,這篇文章介紹的還是很詳細(xì)的,相信本文會(huì)對(duì)大家學(xué)習(xí)Go語(yǔ)言具有一定的參考價(jià)值,如果有疑問(wèn)大家可以留言交流,小編會(huì)盡快給大家回復(fù)的,也請(qǐng)大家繼續(xù)支持服務(wù)器之家。