GO的range具體使用
GO 語言的 for…range 能做什麼呢?
for…range 如何使用 ?
for…range 的返回值有哪些情況,可以對於任何數據結構使用嗎?
for…range 的返回值如果不需要,可以如何處理?
for…range 的數據是如何傳遞的?
剛學習 golang 的 xdm 會不會有上面的疑問?其實很簡單,我們就一個一個的來分享和實操一遍
GO 語言的 for…range 能做什麼呢?
golang 的 for…range 是 go 自身的語法,可以用來遍歷數據結構,有如下數據結構可以遍歷
- 切片 slice
- 數組 array
- map 哈希表
- channel 通道
for…range 如何使用 ?
我們分別來看看可以如何使用他們,for…range 相當於一個迭代器,可以遍歷數據結構的鍵/索引 和值
數組 array
初始化一個數組
使用 for…range 遍歷 , 對應的是 索引 和 值
func main() { myArray := [5]int{1, 2, 3, 4, 5} for i, v := range myArray { fmt.Printf("%d -- %d -- %p\n", i, v, &v) } }
切片 slice
初始化一個切片
使用 for…range 遍歷 , 對應的是 索引 和 值
mySlice := []int{1, 2, 3, 4, 5} for i, v := range mySlice { fmt.Printf("%d -- %d -- %p\n", i, v, &v) }
map 哈希表
- 初始化一個 map 哈希表
- 使用 for…range 遍歷 , map 對應的 鍵值對
myMap := map[string]string{ "name": "xmt", "hobby": "program", "addr": "mengli", } for k, v := range myMap { fmt.Printf("%s -- %s -- %p\n", k, v, &v) }
channel 通道
- 創建一個可以緩沖 10 個 int 類型數據的通道
- 創建一個協程專門向通道中寫入數據
- 主協程遍歷通道,讀取數據
package main import "fmt" var myCh = make(chan int, 10) func writeCh() { for i := 0; i < 5; i++ { myCh <- i } close(myCh) } func main() { go writeCh() for { for data := range myCh { fmt.Println(data) } break } }
for…range 的返回值有哪些情況,可以對於任何數據結構使用嗎?
並不是所有數據結構都可以使用 for…range 的,如下結構可以使用這個方法
返回值 1 | 返回值 2 | 數據傳遞 | |
---|---|---|---|
字符串 | 索引 | 索引對應的值 | 值傳遞 |
數組或者切片 | 索引 | 索引對應的值 | 數組:值傳遞 切片:引用傳遞 |
哈希表 | 鍵 | 鍵對應的值 | 指針 |
通道 | 通道中的數據 | 指針 |
for…range 的返回值如果不需要,可以如何處理?
相信寫過 golang 的 xdm 都知道,go 裡面對於我們不需要的返回值,都可以使用 – 來表示 ,因此 for…range 當然也可以這樣
例如:
myMap := map[string]string{ "name": "xmt", "hobby": "program", "addr": "mengli", } for _, v := range myMap { fmt.Printf("%s -- %p\n", v, &v) }
for…range 的數據是如何傳遞的?
數據都是通過復制傳遞的,也就是都是值傳遞的,隻不過由於數據結構不一樣,數組和切片在這裡還是有一些區別的
前面的文章,我們有說到,切片,對應著 有 3 個元素的數據結構,一個是 cap,一個是 len, 一個是 ptr ,指向的是一個底層數組
切片是引用傳遞,但是傳遞數據的時候,切片這個變量是值傳遞,但是它實際指向的底層數組還是不會變的
咱們來寫一個 demo 查看一下:
我們思路是這個樣子:
遍歷一個 數組/切片,在遍歷當前值的時候,去修改後面索引對應的值,全部遍歷完畢後,查看結果中,實際的數據是否會被修改掉,若修改掉瞭,則是切片是傳引用,若沒有修改掉,則是數組是傳值
數組的效果
myArray := [5]int{1, 2, 3, 4, 5} fmt.Println(myArray) for i, v := range myArray { if i == 0{ myArray[2] = 888 } fmt.Printf("%d -- %d -- %p\n", i, v, &myArray[i]) }
效果如下:
go run main.go
[1 2 3 4 5]
0 — 1 — 0x1189c120
1 — 2 — 0x1189c124
2 — 3 — 0x1189c128
3 — 4 — 0x1189c12c
4 — 5 — 0x1189c130
切片的效果
mySlice := []int{1, 2, 3, 4, 5} fmt.Println(mySlice) for i, v := range mySlice { if i == 0{ mySlice[2] = 888 } fmt.Printf("%d -- %d -- %p\n", i, v, &mySlice[i]) }
效果如下:
go run main.go
[1 2 3 4 5]
0 — 1 — 0x1140e340
1 — 2 — 0x1140e344
2 — 888 — 0x1140e348
3 — 4 — 0x1140e34c
4 — 5 — 0x1140e350
通過上述的案例,相信心裡都有點譜瞭吧
到此這篇關於GO的range具體使用的文章就介紹到這瞭,更多相關GO range內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- golang中for range的取地址操作陷阱介紹
- golang中的空slice案例
- Golang range slice 與range array 之間的區別
- 關於Go 是傳值還是傳引用?
- 詳解Go語言Slice作為函數參數的使用