详解golang defer 闭包 匿名函数

 更新时间:2022年07月14日 09:31:07   作者:@航空母舰  
这篇文章主要介绍了golang defer 闭包 匿名函数的相关知识,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

defer用于资源的释放,会在函数返回之前进行调用。如果有多个defer表达式,调用顺序类似于栈,越后面的defer表达式越先被调用。

defer的触发时机

  • 包裹着defer语句的函数返回时
  • 包裹着defer语句的函数执行到最后时
  • 当前goroutine发生Panic时

当前goroutine发生Panic时

//输出结果:return前执行defer
func f1() {
	defer fmt.Println("return前执行defer")
	return
}
 
//输出结果:函数执行
// 函数执行到最后
func f2() {
	defer fmt.Println("函数执行到最后")
	fmt.Println("函数执行")
}
 
//输出结果:panic前  第一个defer在Panic发生时执行,第二个defer在Panic之后声明,不能执行到
func f3() {
	defer fmt.Println("panic前")
	panic("panic中")
	defer fmt.Println("panic后")
}

defer,return,返回值的执行顺序

  • 先给返回值赋值
  • 执行defer语句
  • 包裹函数return返回
func f1() int { //匿名返回值
	var r int = 6
	defer func() {
		r *= 7
	}()
	return r
}
 
func f2() (r int) { //有名返回值
	defer func() {
		r *= 7
	}()
	return 6
}
 
func f3() (r int) { //有名返回值
	defer func(r int) {
		r *= 7
	}(r)
	return 6
}

f1的执行结果是6, f2的执行结果是42,f3的执行结果是6

最后看example3。它改写后变成

func f1() (r int) {
	r = 6 //给返回值赋值
	func(r int) { //这里改的r是传值传进去的r,不会改变要返回的那个r值
		r *= 7
	}(r)
	return //空的return
}

f1的结果是6。f1是匿名返回值,匿名返回值是在return执行时被声明,因此defer声明时,还不能访问到匿名返回值,defer的修改不会影响到返回值。
f2先给返回值r赋值,r=6,执行defer语句,defer修改r, r = 42,然后函数return。
f3是有名返回值,但是因为r是作为defer的传参,在声明defer的时候,就进行参数拷贝传递,所以defer只会对defer函数的局部参数有影响,不会影响到调用函数的返回值。

闭包与匿名函数

匿名函数:没有函数名的函数。函数也是一种类型,一个函数可以赋值给变量
闭包:可以使用另外一个函数作用域中的变量的函数。闭包复制的是原对象指针,这就很容易解释延迟引用现象。

for i := 0; i <= 3; i++ {
    defer func() {
        fmt.Print(i)
    }
}
//输出结果时 3,3,3,3
因为defer函数的i是对for循环i的引用,defer延迟执行,for循环到最后i是3,到defer执行时i就是3
 
for i := 0; i <= 3; i++ {
    defer func(i int) {
        fmt.Print(i)
    }(i)
}
//输出结果时 3,2,1,0
因为defer函数的i是在defer声明的时候,就当作defer参数传递到defer函数中

匿名函数

func main() {
	/* 匿名函数切片初始化 */
	fns := [](func(x int) int){
		func(x int) int { return x + 1 },
		func(x int) int { return x + 113 },
	}
	println(fns[1](100))
 
	/* 结构体初始化 */
	d := struct {
		fn func() string
	}{
		fn: func() string { return "Hello, World!" },
	}
	println(d.fn())
 
	fc := make(chan func() string, 2)
	fc <- func() string { return "Hello, World!" }
	println((<-fc)())
}

闭包

package main
 
import "fmt"
 
func test() func() {
	x := 100
	fmt.Printf("x (%p) = %d\n", &x, x)
 
	return func() {
		fmt.Printf("x (%p) = %d\n", &x, x)
	}
}
 
func main() {
	f := test()
	f()
}

输出:

x (0xc42007c008) = 100
x (0xc42007c008) = 100

到此这篇关于golang defer 闭包 匿名函数的文章就介绍到这了,更多相关golang defer  匿名函数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • go语言调用c语言的so动态库的实现

    go语言调用c语言的so动态库的实现

    在Go语言开发过程中,有时需要调用C或C++编写的so动态库,本文介绍了如何在Go语言中调用so库的步骤和注意事项,包括环境准备、编译生成.so文件、Go文件编写、以及可能遇到的问题和解决方法,感兴趣的可以了解一下
    2024-10-10
  • 如何编写Go语言中间件的实例教程

    如何编写Go语言中间件的实例教程

    不知道大家有没有写过中间件呢,它是怎么写的呢?下面这篇文中就来给大家分享一下使用Go,如何编写中间件,文中通过示例代码介绍的非常详细,供大家参考学习,下面随着小编来一起学习学习吧。
    2018-04-04
  • GoLang strings.Builder底层实现方法详解

    GoLang strings.Builder底层实现方法详解

    自从学习go一个月以来,我多少使用了一下strings.Builder,略有心得。你也许知道它,特别是你了解bytes.Buffer的话。所以我在此分享一下我的心得,并希望能对你有所帮助
    2022-10-10
  • Go select 死锁的一个细节

    Go select 死锁的一个细节

    这篇文章主要给大家分享的是Go select 死锁的一个细节,文章先是对主题提出问题,然后展开内容,感兴趣的小伙伴可以借鉴一下,希望对你有所帮助
    2021-10-10
  • gin项目部署到服务器并后台启动的步骤

    gin项目部署到服务器并后台启动的步骤

    本文主要介绍了gin项目部署到服务器并后台启动的步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-02-02
  • Go语言基础模板设计模式示例详解

    Go语言基础模板设计模式示例详解

    这篇文章主要为大家介绍了Go语言基础设计模式之模板模式的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2021-11-11
  • 详解Go语言中数组,切片和映射的使用

    详解Go语言中数组,切片和映射的使用

    Arrays (数组), Slices (切片) 和 Maps (映射) 是常见的一类数据结构。这篇文章将为大家详细介绍一下Go语言中数组,切片和映射的使用,感兴趣的可以学习一下
    2022-07-07
  • 手把手教你如何在Goland中创建和运行项目

    手把手教你如何在Goland中创建和运行项目

    欢迎来到本指南!我们将手把手地教您在Goland中如何创建、配置并运行项目,通过简单的步骤,您将迅速上手这款强大的集成开发环境(IDE),轻松实现您的编程梦想,让我们一起开启这段精彩的旅程吧!
    2024-02-02
  • 如何在Go语言中高效使用Redis的Pipeline

    如何在Go语言中高效使用Redis的Pipeline

    在 Redis 中,Pipeline 就像一条流水线,它允许我们将多个命令一次性发送到服务器,下面我们就来看看如何在Go语言中高效使用Redis的Pipeline吧
    2024-11-11
  • Go语言中常用语法编写与优化技巧小结

    Go语言中常用语法编写与优化技巧小结

    为了充分利用 Go 的潜力,我们需要了解如何优化 Go 程序,本文将介绍一些常见的 Go 语言优化技巧,并通过实际例子进行说明,希望对大家有所帮助
    2024-02-02

最新评论