不破楼兰终不还——Go 延迟语句defer指南

2023-11-10 01:30

本文主要是介绍不破楼兰终不还——Go 延迟语句defer指南,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

不破楼兰终不还——Go 延迟语句defer指南

说到defer,很多gopher都知道这是求职面试常考点,也是一个易错的难点,特别是延迟语句defer也是Golang一个十分重要的关键字。所以掌握defer刻不容缓!

什么是defer?

现在我们编程经常要操作文件或数据库,而进行数据库和文件操作就会涉及数据库和文件的关闭,用完不关闭就会导致内存泄露,可能会导致很严重的安全问题。但是这也是我们经常忘记的一个步骤,所以defer可以很好的解决这个问题,比如我们连接数据库就使用defer编写关闭语句,这就能较好的帮助开发人员编写更安全的程序。

不过defer运行时会带来一定时间的开销,因此如果对耗时要求特别严格,建议不使用defer。

我们来看下面这个例子:

var l sync.RWMutex
l.Lock()
panic("异常信息")
l.Unlock()

如果上面这种情况,使用锁和解锁语句之间出现了panic,就会形成死锁,即使我们不使用panic语句,其他语句也可能导致panic啊,因此我们需要使用defer。

image-20220827104219450

defer的执行顺序是什么?

根据Golang官方文档描述,defer就像一个LIFO的栈,每次执行defer语句,都会将函数”压栈“,函数参数也会被保存下来;如果外层函数(非代码块)退出,最后的defer语句就会执行,也就是栈顶的函数或方法会被执行。

不过需要注意:

如果defer执行的语句是一个nil,那么就会在调用时产生panic。

一般情况下多个defer的执行顺序,我们可以通过下面这个例子了解:

package mainimport ("fmt"
)func main() {defer fmt.Println("defer 1")defer_test()defer fmt.Println("defer 2")
}func defer_test() {defer fmt.Println("defer 3")defer fmt.Println("defer 4")
}

运行结果为:

defer 4
defer 3
defer 2
defer 1Program exited.

defer就像一个LIFO的栈,最后被定义的defer最先执行。

image-20220827104251903

参数传递无外乎就是传值(pass by value),传引用(pass by reference)或者说是传指针。在Go语言中,按引用传递其实也可以称作”按值传递”,只不过该副本是一个地址的拷贝,通过它可以修改这个值所指向的地址上的值。

使用defer时,涉及到函数参数和闭包引用。使用函数参数方式,defer会在定义时取值并保存起来。而使用闭包引用的方式,虽然也是值传递,但是拷贝的是函数指针。

举个栗子:

package mainimport ("fmt"
)func main() {var array [5]int = [5]int{5, 4, 3, 2, 1}for _, i := range array {defer func() { fmt.Println(i) }()}
}

运行结果如下:

1
1
1
1
1Program exited.

defer语句全是输出1,因为循环结束后i=1,而使用匿名函数让defer后面跟着的是一个“闭包”,所以i是“引用类型”的变量。

如果对上面这段代码稍作修改,得到的结果就不一样了:

package mainimport ("fmt"
)func main() {var array [5]int = [5]int{5, 4, 3, 2, 1}for _, i := range array {defer fmt.Println(i)}
}

运行结果如下:

1
2
3
4
5Program exited.

调用 defer 关键字会立刻拷贝函数中引用的外部参数,因此当i从5到1时,所有的值都被拷贝下来。

image-20220827104030980

defer与return的执行顺序

defer用得好则已,用得不好就会带来灾难。

能不能用好就得看我们能不能理解retrun语句。

一条return语句,其实不是一条原子指令,其大概可以分为三条指令:

  • 返回值为xxx
  • 调用defer函数
  • 空的return

举个栗子:

package mainimport ("fmt"
)func main() {fmt.Println("return:", banana()) 
}func banana() (i int) {defer func() {i++fmt.Println("defer 2:", i) }()defer func() {i++fmt.Println("defer 1:", i) }()return i 
}
defer 1: 1
defer 2: 2
return: 2Program exited.

这是有名返回值的情况,接下我们来看一看匿名返回值的情况:

package mainimport ("fmt"
)func main() {fmt.Println("return:", apple())
}func apple() int {var i intdefer func() {i++fmt.Println("defer 2:", i)}()defer func() {i++fmt.Println("defer 1:", i)}()return i
}
defer 1: 1
defer 2: 2
return: 0Program exited.

上面这两段代码说明了:defer语句只能访问有名返回值,不能直接访问匿名返回值。

但是如果是下面这种情况:

package mainimport ("fmt"
)func main() {fmt.Println("return:", banana())
}func banana() (i int) {defer func(i int) {i++fmt.Println("defer 2:", i)}(i)defer func(i int) {i++fmt.Println("defer 1:", i)}(i)return i
}

输出结果就为:

defer 1: 1
defer 2: 1
return: 0Program exited.

这是因为传递给defer后面的匿名函数的是形参的一个复制值,不会影响实参i。

参考文献

机械工业出版社 《Go程序员面试笔试宝典》

defer的执行顺序与时机 https://studygolang.com/articles/22931

理解 Go 语言 defer 关键字的原理 | Go 语言设计与实现

https://draveness.me/golang/docs/part2-foundation/ch05-keyword/golang-defer/#531-现象

GO函数传参 []int 与 [3]int 有何区别?https://segmentfault.com/q/1010000020543158?bd_source_light=4746641

Golang中defer、return、返回值之间执行顺序的坑 https://cloud.tencent.com/developer/article/1410243

这篇关于不破楼兰终不还——Go 延迟语句defer指南的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/379641

相关文章

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

使用Java填充Word模板的操作指南

《使用Java填充Word模板的操作指南》本文介绍了Java填充Word模板的实现方法,包括文本、列表和复选框的填充,首先通过Word域功能设置模板变量,然后使用poi-tl、aspose-words... 目录前言一、设置word模板普通字段列表字段复选框二、代码1. 引入POM2. 模板放入项目3.代码

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc

macOS彻底卸载Python的超完整指南(推荐!)

《macOS彻底卸载Python的超完整指南(推荐!)》随着python解释器的不断更新升级和项目开发需要,有时候会需要升级或者降级系统中的python的版本,系统中留存的Pytho版本如果没有卸载干... 目录MACOS 彻底卸载 python 的完整指南重要警告卸载前检查卸载方法(按安装方式)1. 卸载

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

Python动态处理文件编码的完整指南

《Python动态处理文件编码的完整指南》在Python文件处理的高级应用中,我们经常会遇到需要动态处理文件编码的场景,本文将深入探讨Python中动态处理文件编码的技术,有需要的小伙伴可以了解下... 目录引言一、理解python的文件编码体系1.1 Python的IO层次结构1.2 编码问题的常见场景二

Oracle Scheduler任务故障诊断方法实战指南

《OracleScheduler任务故障诊断方法实战指南》Oracle数据库作为企业级应用中最常用的关系型数据库管理系统之一,偶尔会遇到各种故障和问题,:本文主要介绍OracleSchedul... 目录前言一、故障场景:当定时任务突然“消失”二、基础环境诊断:搭建“全局视角”1. 数据库实例与PDB状态2

Git进行版本控制的实战指南

《Git进行版本控制的实战指南》Git是一种分布式版本控制系统,广泛应用于软件开发中,它可以记录和管理项目的历史修改,并支持多人协作开发,通过Git,开发者可以轻松地跟踪代码变更、合并分支、回退版本等... 目录一、Git核心概念解析二、环境搭建与配置1. 安装Git(Windows示例)2. 基础配置(必