写点什么

Go 语言学习查缺补漏 ing Day1

  • 2021 年 12 月 02 日
  • 本文字数:1467 字

    阅读完需:约 5 分钟

作者:ReganYue

来源:恒生LIGHT云社区

Go 语言学习查缺补漏 ing Day1

零、前言

因为笔者基础不牢,在使用 Go 语言的时候经常遇到很多摸不着头脑的问题,所以笔者下定决心好好对 Go 语言进行查漏补缺,本【Go 语言查缺补漏 ing】系列主要是帮助新手 Gopher 更好的了解 Go 语言的易错点、重难点。希望各位看官能够喜欢,点点赞、关注一下呗!

一、多个 defer 的执行顺序

package main
import "fmt"
func main() { defer fmt.Println("天才第一步") defer fmt.Println("雀氏纸尿裤") defer fmt.Println("战神第一步") defer fmt.Println("盖亚纸尿裤")
}
复制代码


我们在实际项目开发中,经常会遇到使用多个 defer 来进行延时处理的情况,这时候,了解多个 defer 同时存在的情况下,它们的执行顺序就是十分重要的了。


这段代码的运行结果是:


盖亚纸尿裤战神第一步雀氏纸尿裤天才第一步
复制代码


也就是说,当多个 defer 语句并称时,它们遵循后进先出的顺序。

二、defer 与 return 的先后执行顺序

package main
import "fmt"
func main() { fmt.Println("主函数:", d())}func d() int { i:=0 defer func() { i+=10 fmt.Println("先来的defer语句:", i) }() defer func() { i++ fmt.Println("后到的defer语句:", i) }() return i}
复制代码


先看一看执行结果:


后到的defer语句: 1先来的defer语句: 11主函数: 0
复制代码


明白执行顺序了吗?


明白的同学这里可以跳过了,我来讲解一下,前面讲了 defer 语句的执行顺序是后到先出。所以两个 defer 中先输出后到 defer 语句,再输出先来的 defer 语句。但是主函数输出 0,这就说明返回值是 0,这就说明是先 return 再执行 defer 语句。并且注意函数结束是 defer 完成之后再结束。

三、for...range...创建的是每个元素的副本

package main
import "fmt"
func main() { slice := []int{0, 1, 2, 3, 4, 5} m := make(map[int]*int) for key, value := range slice { m[key] = &value }
for k, v := range m { fmt.Println("key=",k,"value=",*v) }}
复制代码


先来看一看运行结果,再来具体讲述:


key= 0 value= 5key= 1 value= 5key= 2 value= 5key= 3 value= 5key= 4 value= 5key= 5 value= 5
复制代码


我们会发现 key 没问题,但是 value 都是一样的,是不是明白了什么?m[key] = &value是不是取得是 value 的地址,这就说明 for...range...生成的是每个元素的副本,而不是每个元素的引用。至于为什么都是 5,而不是其它,是因为 value 最后被赋值 5.而所有 value 都指向这个地址,所以输出的所有 value 都相同。


我们可以加一行代码,就能达到我们本来想达到的目的了。


package main
import "fmt"
func main() { slice := []int{0, 1, 2, 3, 4, 5} m := make(map[int]*int) for key, value := range slice { v := value m[key] = &v }
for k, v := range m { fmt.Println("key=",k,"value=",*v) }}
复制代码


修改后的程序运行结果是:


key= 3 value= 3key= 4 value= 4key= 5 value= 5key= 0 value= 0key= 1 value= 1key= 2 value= 2
复制代码

四、用 make 创建 slice 需要注意的一处小地方

package main
import "fmt"
func main() { s1 := make([]int,3) s2 := make([]int,0) s1 = append(s1, 8, 8, 8) s2 = append(s2, 8, 8, 8) fmt.Println("s1=>",s1) fmt.Println("s2=>",s2)}
复制代码


运行结果是:


s1=> [0 0 0 8 8 8]s2=> [8 8 8]
复制代码


我们可以看到,append 了三个 8 之后,s1 前面有 3 个 0,而 s2 前面却没有 0。这就说明使用 make 创建 slice,会自动填充 n 个零。

发布于: 3 小时前阅读数: 6
用户头像

还未添加个人签名 2018.11.07 加入

还未添加个人简介

评论

发布
暂无评论
Go语言学习查缺补漏ing Day1