不破楼兰终不还——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指南的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/qq_36045898/article/details/126555392
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/379641

相关文章

Apache 高级配置实战之从连接保持到日志分析的完整指南

《Apache高级配置实战之从连接保持到日志分析的完整指南》本文带你从连接保持优化开始,一路走到访问控制和日志管理,最后用AWStats来分析网站数据,对Apache配置日志分析相关知识感兴趣的朋友... 目录Apache 高级配置实战:从连接保持到日志分析的完整指南前言 一、Apache 连接保持 - 性

Mybatis Plus JSqlParser解析sql语句及JSqlParser安装步骤

《MybatisPlusJSqlParser解析sql语句及JSqlParser安装步骤》JSqlParser是一个用于解析SQL语句的Java库,它可以将SQL语句解析为一个Java对象树,允许... 目录【一】jsqlParser 是什么【二】JSqlParser 的安装步骤【三】使用场景【1】sql语

Nacos日志与Raft的数据清理指南

《Nacos日志与Raft的数据清理指南》随着运行时间的增长,Nacos的日志文件(logs/)和Raft持久化数据(data/protocol/raft/)可能会占用大量磁盘空间,影响系统稳定性,本... 目录引言1. Nacos 日志文件(logs/ 目录)清理1.1 日志文件的作用1.2 是否可以删除

Go语言中使用JWT进行身份验证的几种方式

《Go语言中使用JWT进行身份验证的几种方式》本文主要介绍了Go语言中使用JWT进行身份验证的几种方式,包括dgrijalva/jwt-go、golang-jwt/jwt、lestrrat-go/jw... 目录简介1. github.com/dgrijalva/jwt-go安装:使用示例:解释:2. gi

golang实现延迟队列(delay queue)的两种实现

《golang实现延迟队列(delayqueue)的两种实现》本文主要介绍了golang实现延迟队列(delayqueue)的两种实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目录1 延迟队列:邮件提醒、订单自动取消2 实现2.1 simplChina编程e简单版:go自带的time

Python FastAPI实现JWT校验的完整指南

《PythonFastAPI实现JWT校验的完整指南》在现代Web开发中,构建安全的API接口是开发者必须面对的核心挑战之一,本文将深入探讨如何基于FastAPI实现JWT(JSONWebToken... 目录一、JWT认证的核心原理二、项目初始化与环境配置三、安全密码处理机制四、JWT令牌的生成与验证五、

go rate 原生标准限速库的使用

《gorate原生标准限速库的使用》本文主要介绍了Go标准库golang.org/x/time/rate实现限流,采用令牌桶算法控制请求速率,提供Allow/Reserve/Wait方法,具有一定... 目录介绍安装API介绍rate.NewLimiter:创建限流器limiter.Allow():请求是否

JavaScript实战:智能密码生成器开发指南

本文通过JavaScript实战开发智能密码生成器,详解如何运用crypto.getRandomValues实现加密级随机密码生成,包含多字符组合、安全强度可视化、易混淆字符排除等企业级功能。学习密码强度检测算法与信息熵计算原理,获取可直接嵌入项目的完整代码,提升Web应用的安全开发能力 目录

Linux网络配置之网桥和虚拟网络的配置指南

《Linux网络配置之网桥和虚拟网络的配置指南》这篇文章主要为大家详细介绍了Linux中配置网桥和虚拟网络的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、网桥的配置在linux系统中配置一个新的网桥主要涉及以下几个步骤:1.为yum仓库做准备,安装组件epel-re

Go 语言中的 Struct Tag 的用法详解

《Go语言中的StructTag的用法详解》在Go语言中,结构体字段标签(StructTag)是一种用于给字段添加元信息(metadata)的机制,常用于序列化(如JSON、XML)、ORM映... 目录一、结构体标签的基本语法二、json:"token"的具体含义三、常见的标签格式变体四、使用示例五、使用