我是靠谱客的博主 兴奋百褶裙,最近开发中收集的这篇文章主要介绍三分钟学 Go 语言——range深度解析,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

小熊最近两天加班比较严重,要处理的事情很多,但是学习的热情永远不会减少,前面讲述的go语言语法是非常非常简单的,所以没有做深入的剖析,后面会从各种角度解析语法,fighting!!

range(范围)

range 关键字在 go 语言中是相当常用好用的语法糖,可以用在 for 循环中迭代 arrayslicemapchannel字符串所有涉及到遍历输出的东西。

基本原理
怎么用?

我们在前一节循环中初次触及到了 range,也知道他可以省略key,或者省略value来循环遍历的特性,但是这种特性要结合实际情况考量该用哪一个。

切片迭代

	nums := []int{1, 2, 3}
	for k, v := range nums {
		fmt.Printf("key: %v , value: %v  n", k, v)
	}

这和迭代方式非常适合用for-range语句,如果减少赋值,直接索引num[key]可以减少损耗。如下

for k, _:= range nums {

map迭代
注意,从 Go1开始,遍历的起始节点就是随机了。

	kvs := map[string]string{
		"a":"a",
		"b":"b",
	}
	for k, v := range kvs {
		fmt.Printf("key: %v , value: %v  n", k, v)
	}

函数中for-range语句中只获取 key 值,然后跟据 key 值获取 value 值,虽然看似减少了一次赋值,但通过 key 值查找 value 值的性能消耗可能高于赋值消耗。

所以能否优化取决于 map 所存储数据结构特征、结合实际情况进行。

字符串迭代(一个一个的输出字符)

for k,v := range "hello"{
  //注意这里单个字符输出的是ASCII码,
  //用 %c 代表输出字符
		fmt.Printf("key: %v , value: %c
    n", k, v)
	}

channel (如果不会可以先 mark 下,详细参考后续:go 的并发特性章节)

ch := make(chan int, 10)
	ch <- 11
	ch <- 12

	close(ch) // 不用的时候记得关掉,不关掉又没有另一个goroutine存在会死锁哦,可以注释掉这一句体验死锁

	for x := range ch {
		fmt.Println(x)
	}

结构体

tmp := []struct{
		int
		string
	}{
		{1, "a"},
		{2, "b"},
	}

	for k,v := range tmp{
		fmt.Printf("k:%v, v:%v  n",k,v)
	}

注意:由于循环开始前循环次数就已经确定了,所以循环过程中新添加的元素是没办法遍历到的。

有可能会遇到的坑!

由于range遍历时value是值的拷贝,如果这个时候遍历上一节声明的结构体时,修改value,原结构体不会发生任何变化!

for _,v := range tmp{
		v.a = 2
	}

两次输出一致

k:0, v:{1 a}  
k:1, v:{2 b}  
k:0, v:{1 a}  
k:1, v:{2 b}  
编程 Tips
  • 遍历过程中可以适情况放弃接收 indexvalue,可以一定程度上提升性能
  • 遍历 channel 时,如果 channel 中没有数据,可能会阻塞
  • 尽量避免遍历过程中修改原数据
总结
  • range可以用于for 循环中迭代 arrayslicemapchannel字符串所有涉及到遍历输出的东西。
  • for-range 的实现实际上是C风格的for循环
  • 使用index,value接收range返回值会发生一次数据拷贝

最后

以上就是兴奋百褶裙为你收集整理的三分钟学 Go 语言——range深度解析的全部内容,希望文章能够帮你解决三分钟学 Go 语言——range深度解析所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(37)

评论列表共有 0 条评论

立即
投稿
返回
顶部