Go函数下篇:defer和闭包

简介: Go函数下篇:defer和闭包

defer

  1. 使用defer注册的匿名函数(延迟调用)还可以给它传参,不过是值拷贝
package main
import "fmt"
func work() int {
 num := 10
 defer func(i int) {
  i += 20
  println("defer内的结果:", i)
 }(num)
 return num
}
func main() {
 ret := work()
 fmt.Println(ret)
}

输出:

defer内的结果: 30
10

解析:在work函数内,变量num传递给了通过defer注册了的匿名函数,在匿名函数内做了加20,但它的结果并未影响到外面的num变量,这就是值拷贝。

  1. 当主动调用os.Exit退出程序后,defer会不会执行?试试再说
package main
import (
 "fmt"
 "os"
)
func main() {
 defer func() {
  fmt.Println("释放锁...")
 }()
 fmt.Println("hello")
 os.Exit(1)
}

输出:

hello

答案很明显了,居然不会。

  1. 注册自定义的函数为延迟调用,模拟个小栗子:打开文件,处理,处理完后关闭该文件
package main
import (
 "fmt"
)
func closeFile() {
 println("关闭文件")
}
func work() bool {
 defer closeFile()
 fmt.Println("打开文件")
 fmt.Println("正在处理...")
 fmt.Println("处理完成!")
 return true
}
func main() {
 status := work()
 fmt.Println(status)
}

输出:

打开文件
正在处理...
处理完成!
true
关闭文件

所以说,defer的应用场景可以是释放资源或者关闭连接、关闭文件、释放锁等等,说白了就是:它就是做收尾工作的家伙。

闭包

  1. 闭包初体验

闭包的本质其实返回的是一个函数,但这个函数有点特殊,为啥说特殊呢?因为在这个函数里面,还有一个函数,这个函数是匿名函数,且在这个匿名函数里面还可以引用外部变量,当被反复调用时,这里引用的外部变量只会初始化一次。因此:闭包=函数+函数里面的匿名函数+匿名函数引用的外部变

「接下来一步一步进行解剖」

刚才说到闭包的本质其实返回的是一个函数,之前提到过,匿名函数是可以作为函数的返回值的,看下面代码:

func bibao() func() {
}

上面的栗子中,定义了一个函数Bibao,这个函数没有入参,返回值的类型是函数类型,也就是定义成func(),所以它要返回一个函数才行呢,现在函数体是空的,别急,一步一步解剖。

刚才说到在这个函数里面,还有一个函数,这个函数是匿名函数,且还要返回这个匿名函数,看下面的代码:

func bibao() func() {
 return func() {
  fmt.Println("hello")
 }
}
func main() {
 b := bibao()
 b()
}

输出:

hello

在bibao函数体里,返回了一个匿名函数,这个匿名函数的功能是输出了字符串“hello”,调用bibao()把它赋值给变量b,此刻,变量b就是一个函数(它接收的就是返回的匿名函数),变量b既然是函数,那就可以进行调用了b(),所以执行后,输出的是“hello”。

刚才说到且在这个匿名函数里面还可以引用外部变量,当被反复调用时,这里引用的外部变量只会初始化一次,看下面代码:

package main
import "fmt"
func bibao(name string) func() string {
 base := "hello"
 return func() string {
  base = base + " " + name
  return base
 }
}
func main() {
 b := bibao("ttr")
 fmt.Println(b()) // 第1次调用
 fmt.Println(b()) // 第2次调用
 fmt.Println(b()) // 第3次调用
 fmt.Println(b()) // 第4次调用
}

输出:

hello ttr //第1次调用的结果输出
hello ttr ttr //第2次调用的结果输出
hello ttr ttr ttr //第3次调用的结果输出
hello ttr ttr ttr ttr //第4次调用的结果输出

这次对bibao函数做了点小改造,它需要接收一个参数name。在bibao函数中,对于匿名函数内部来说,它的外部变量就是bibao函数里的base变量。调用函数bibao并传入"ttr",在匿名函数中进行了字符串拼接,bibao("ttr")初始化了一次,b()反复调用了4次,在匿名函数中引用的外部变量base,每次调用时返回的是外部变量base的多个副本,因为在每次调用时都会为局部变量分配内存(对于整个程序来说,在函数里的base变量是局部变量,而对于在bibao函数里的匿名函数来说,base变量是外部变量。)

还可以再这样改造,效果也是一样的:

package main
import "fmt"
func bibao() func(name string) string {
 base := "hello"
 return func(name string) string {
  base = base + " " + name
  return base
 }
}
func main() {
 b := bibao()
 fmt.Println(b("ttr"))
 fmt.Println(b("ttr1"))
 fmt.Println(b("ttr2"))
}

输出:

hello ttr
hello ttr ttr1
hello ttr ttr1 ttr2

好了,到此为止,闭包到底在哪呢?下面这块就是闭包,bibao函数返回了这个闭包:

base := "hello"
return func(name string) string {
 base = base + " " + name
 return base
}

闭包是由函数和与其相关的引用环境组合而成的实体,这里所说的函数就是匿名函数,所说的引用环境就是外部变量base,他们一起组合成了一个实体并返回,这就是闭包。

突然想到如果把base变量放到匿名函数里面,它还是不是一个闭包?看看效果:

package main
import "fmt"
func bibao() func(name string) string {
 return func(name string) string {
  base := "hello "
  base = base + " " + a + " " + name
  return base
 }
}
func main() {
 b := bibao()
 fmt.Println(b("ttr"))
 fmt.Println(b("ttr1"))
 fmt.Println(b("ttr2"))
}

输出:

hello ttr
hello ttr1
hello ttr2
  1. 来一个小栗子

闭包的好处在于,可以减少全局变量,在函数调用的过程中隐式的传递共享变量,这更具有封装性,更安全了,使其不能随意访问到共享变量。

package main
import (
 "fmt"
)
func bankcard(name string) func() (string, float32) {
 amount := 10000.0
 return func() (string, float32) {
  return name, float32(amount)
 }
}
func main() {
 b := bankcard("xiaoming")
 fmt.Println(b())
}
相关文章
|
1天前
|
Go
Go语言中defer的执行顺序详解
【2月更文挑战第22天】
29 4
|
1天前
|
Go 数据处理
Go杂记1-切片Slice作为函数参数那点事儿
Go杂记1-切片Slice作为函数参数那点事儿
7 0
|
1天前
|
Go C++
go 语言回调函数和闭包
go 语言回调函数和闭包
|
1天前
|
Java Go 区块链
【Go语言专栏】Go语言中的延迟执行与defer语句
【4月更文挑战第30天】Go语言的延迟执行与defer语句用于资源释放和错误处理。defer通过关键字定义,函数返回时执行,顺序与定义相反。参数在定义时求值。应用包括资源释放、错误处理、成对操作和函数包装,是Go编程的关键特性。
|
1天前
|
存储 Go 开发者
【Go语言专栏】函数在Go语言中的使用与实现
【4月更文挑战第30天】本文介绍了Go语言中函数的使用和实现,包括函数定义、参数传递、返回值、匿名函数、变长参数、函数类型、闭包和错误处理。通过示例展示了如何定义和调用函数,以及如何利用闭包和递归解决问题。此外,还提到了Go函数作为一等公民的特性,允许存储和传递。进一步学习可参考官方文档和相关书籍。
|
1天前
|
Go 开发者
Golang深入浅出之-Go语言 defer、panic、recover:异常处理机制
Go语言中的`defer`、`panic`和`recover`提供了一套独特的异常处理方式。`defer`用于延迟函数调用,在返回前执行,常用于资源释放。它遵循后进先出原则。`panic`触发运行时错误,中断函数执行,直到遇到`recover`或程序结束。`recover`在`defer`中捕获`panic`,恢复程序执行。注意避免滥用`defer`影响性能,不应对可处理错误随意使用`panic`,且`recover`不能跨goroutine捕获panic。理解并恰当使用这些机制能提高代码健壮性和稳定性。
26 2
|
1天前
|
Go
Golang深入浅出之-Go语言函数基础:定义、调用与多返回值
【4月更文挑战第21天】Go语言函数是代码组织的基本单元,用于封装可重用逻辑。本文介绍了函数定义(包括基本形式、命名、参数列表和多返回值)、调用以及匿名函数与闭包。在函数定义时,注意参数命名和注释,避免参数顺序混淆。在调用时,要检查并处理多返回值中的错误。理解闭包原理,小心处理外部变量引用,以提升代码质量和可维护性。通过实践和示例,能更好地掌握Go语言函数。
30 1
Golang深入浅出之-Go语言函数基础:定义、调用与多返回值
|
1天前
|
程序员 Go API
【Go语言快速上手(二)】 分支与循环&函数讲解
【Go语言快速上手(二)】 分支与循环&函数讲解
|
1天前
|
存储 Go 开发者
Golang深入浅出之-Go语言字符串操作:常见函数与面试示例
【4月更文挑战第20天】Go语言字符串是不可变的字节序列,采用UTF-8编码。本文介绍了字符串基础,如拼接(`+`或`fmt.Sprintf()`)、长度与索引、切片、查找与替换(`strings`包)以及转换与修剪。常见问题包括字符串不可变性、UTF-8编码处理、切片与容量以及查找与替换的边界条件。通过理解和实践这些函数及注意事项,能提升Go语言编程能力。
28 0
|
1天前
|
自然语言处理 数据挖掘 程序员
《Go 简易速速上手小册》第2章:控制结构与函数(2024 最新版)(下)
《Go 简易速速上手小册》第2章:控制结构与函数(2024 最新版)(上)
27 1

热门文章

最新文章