golang的defer踩坑汇总
原文链接:http://www.zhoubotong.site/post/50.html
defer语句用于延迟函数调用,每次会把一个函数压入栈中,函数返回前再把延迟的函数取出并执行。延迟函数可以有参数:
延迟函数的参数在defer语句出现时就已确定下来(传值的就是当前值)
return先赋值(对于命名返回值),然后执行defer,最后函数返回
延迟函数执行按后进先出顺序执行
延迟函数可操作主函数的变量名返回值(修改返回值)
defer后面的表达式可以是func或者是method的调用,如果defer的函数为nil,则会panic
日常开发中,使用不当很容易造成意外的“坑”。下面我整理了下常规使用场景下,defer的问题可能的踩坑汇总。
释放资源
defer语句用于延迟函数调用,每次会把一个函数压入栈中,函数返回前再把延迟的函数取出并执行。延迟函数可以有参数:
延迟函数的参数在defer语句出现时就已确定下来(传值的就是当前值);
延迟函数执行按后进先出顺序执行;
延迟函数可操作主函数的具名返回值(修改返回值);
defer 语句正好是在函数退出时执行的语句,所以使用 defer 能非常方便地处理资源释放、句柄关闭等问题。
package main
import (
"fmt"
"os"
)
func fileSize(filename string) int64 {
f, err := os.Open(filename)
if err != nil {
return 0
}
// 延迟调用Close, 此时Close不会被调用
defer f.Close()
info, err := f.Stat()
if err != nil {
// defer机制触发, 调用Close关闭文件
return 0
}
size := info.Size()
// defer机制触发, 调用Close关闭文件
return size
}
func main() {
fmt.Println(fileSize("demo.txt"))
}
变量捕获
defer中的变量会被提前捕获,后续的修改不会影响到已捕获的值,举个例子:
package main
import (
"fmt"
)
func main() {
i := 0
defer fmt.Println("Defer运行值:", i)
i = 10 // 这里虽然修改了值,但是不会影响上面的i值
fmt.Println("最后输出值:", i)
}
结果defer语句中打印的值是修改前的值。:
最后输出值: 10
Defer运行值: 0
变量名返回值
在defer中修改具体变量名返回值时,会影响到函数的实际返回值,继续举个例子:
package main import (
"fmt"
) func ShowDefer() {
fmt.Println("最后输出值:", deferValue())
}
func deferValue() (ret int) { // 注意这里返回res变量值
ret = 0 defer func() { // 会直接修改栈中对应的返回值
ret += 10
fmt.Println("Defer 运行值:", ret)
}()
ret = 2
fmt.Println("Ret重置值:", ret)
return //返回的ret最后是 其实是本次2+上面的ret+=10的结果
} func main() {
ShowDefer()
} //Ret重置值: 2
//Defer 运行值: 12
//最后输出值: 12
非变量名返回值
当函数为非具体名返回值时,defer无法影响返回值(因在return时,对应返回值已存入栈中),继续举个例子:
package main import (
"fmt"
) func ShowDefer() {
fmt.Println("最后输出值:", deferValue())
}
func deferValue() int { // 非命名变量返回
ret := 0
defer func() {
ret += 10
fmt.Println("Defer 运行值:", ret)
}()
ret = 2
return ret // 这里直接返回ret2
} func main() {
ShowDefer()
} //Defer 运行值: 12
//最后输出值: 2
经过上面的实践理解,我们来看下下面的笔试题:
笔试题一
package main
import "fmt"
func f() (result int) {
defer func() {
result *= 7
}()
return 3
}
func main() {
fmt.Println(f())
}
问题解析:这里return先给result赋值为3,之后执行defer,result变为21,最后返回21。
笔试题二
package main
import "fmt"
func f() int {
result := 3
defer func() {
result *= 7
}()
return result
}
func main() {
fmt.Println(f())
}
问题解析:这里return确定返回值3,之后defer才修改result,最后函数返回return确定的返回值3。
笔试题三
package main import "fmt"
// 多个defer
func multiDefer() {
for i := 3; i > 0; i-- {
defer func(n int) {
fmt.Print(n, " ")
}(i)
} for i := 33; i > 30; i-- {
defer fmt.Print(i, " ")
}
} func main() {
multiDefer()
}
问题解析:多个defer函数,按顺序逆序执行,这里输出31 32 33 1 2 3 。
笔试题四
package main
import "fmt"
var fun func() string
func main() {
fmt.Println("hello monkey")
defer fun()
}
问题解析:由于这里的defer指定的func为nil,所以会panic 。
笔试题五
package main
import "fmt"
func main() {
for i := 3; i > 0; i-- {
defer func() {
fmt.Print(i, " ")
}()
}
}
问题解析:这里是极度容易踩坑的地方,由于defer这里调用的func没有参数,等执行的时候,i已经为0(按3 2 1逆序,最后一个i=1时,i--的结果最后是0),所以这里输出3个0 。
如果还不太好理解?
package main
import "fmt"
func main() {
for i := 3; i > 1; i-- { // 循环满足条件的是 3 2,
defer func() { // 因为func 没有参数,defer运行最后i--即 2-- 结果为 1
fmt.Print(i, " ") // 循环2次 结果均为 1
}()
}
}//输出 1 1
按照常规的思维理解应该是这样:
package main
import "fmt"
func main() {
for i := 3; i > 0; i-- {
defer func(i int) {
fmt.Print(i, " ")
}(i)
}
}
感兴趣的朋友可以细细品下。
golang的defer踩坑汇总的更多相关文章
- golang协程踩坑记录
1.主线程等待多个协程执行完毕后,再执行下面的程序.golang提供了一个很好用的工具. sync.WaitGroup下面是个简单的例子. 执行结果: 2.主线程主动去结束已经启动了的多个协程.执行结 ...
- Hyperledger Fabric 踩坑汇总
搭建基础环境 阿里云安装出现的一些问题解决 1. [signal SIGSEGV: segmentation violation code=0x1 addr=xxx pc=xxx] 类似的错误:原始错 ...
- vscode 踩坑汇总
gopls 提示 update 将 "go.useLanguageServer": true 改为 "go.useLanguageServer": false
- 微信小程序踩坑集合
1:官方工具:https://mp.weixin.qq.com/debug/w ... tml?t=1476434678461 2:简易教程:https://mp.weixin.qq.com/debu ...
- golang 学习过程中踩的坑
目录 [他人总结] 首字母大写才是对外可见的 包的初始化函数顺序问题 DB 连接泄漏问题 err 常用写法 goroutine 内的变量 指针可能是 nil 多层 map 未初始化 [他人总结] ht ...
- navicate 连接mysql8.0,个人踩坑问题汇总
navicate 连接mysql8.0,个人踩坑问题汇总本文目录:1:安装mysql8.0新增全新验证方式,安装如果不修改mysql连接不上2:mysql启动命令问题3:navicate 运程连接My ...
- Nuxt.js的踩坑指南(常见问题汇总)
本文会不定期更新在nuxt.js中遇到的问题进行汇总.转发请注明出处,尊重作者,谢谢! 强烈推荐作者文档版踩坑指南,点击跳转踩坑指南 在Nuxt的官方文档中,中文文档和英文文档都存在着不小的差异. 1 ...
- Spark踩坑记——共享变量
[TOC] 前言 Spark踩坑记--初试 Spark踩坑记--数据库(Hbase+Mysql) Spark踩坑记--Spark Streaming+kafka应用及调优 在前面总结的几篇spark踩 ...
- Spark踩坑记——从RDD看集群调度
[TOC] 前言 在Spark的使用中,性能的调优配置过程中,查阅了很多资料,之前自己总结过两篇小博文Spark踩坑记--初试和Spark踩坑记--数据库(Hbase+Mysql),第一篇概况的归纳了 ...
随机推荐
- JDK1.8.0_181的无限制强度加密策略文件变动(转载)
JDK1.8.0_181的无限制强度加密策略文件变动 原文地址 https://my.oschina.net/my1313677/blog/3109613 作者 葉者 日常记录 2019/09/23 ...
- POP3协议(电子邮件邮局协议)中UIDL和TOP命令在实际使用中的作用
POP3是电子邮件协议中用于接收邮件的协议,相较于发送邮件的SMTP协议,POP3的命令要多一些.主要的命令有LIST.STAT.RETR.DELE.UIDL.TOP.QUIT,以及用于登录邮箱的US ...
- Transactional事务,事务嵌套的时候,如果主事务出现问题,子事务执行不需要回滚怎么做?
如果调用的方法在不在同一个service当中,则只需要在子事务当中的方法上方添加注解即可 下方即是:这就话代表:重新开启一个新的事务 @Transactional(propagation = Prop ...
- android软件简约记账app开发day05-记账页面条目代码优化和bug解决
android软件简约记账app开发day05-记账页面条目代码优化和bug解决 今天还是因为该bug又极大的耽误了项目进程,该开发文档都要没有时间来写了. 先说bug吧,在昨天已经实现了页面图标的展 ...
- 2021.11.10 [POI2000]病毒(AC自动机)
2021.11.10 [POI2000]病毒(AC自动机) https://www.luogu.com.cn/problem/P2444 题意: 二进制病毒审查委员会最近发现了如下的规律:某些确定的二 ...
- Jenkins+Allure测试报告+飞书机器人发送通知
一.前言 之前讲了jenkins如何设置定时任务执行脚本,结合实际情况,本篇讲述在jenkins构建成功后,如何生成测试报告,以及推送飞书(因为我公司用的是飞书,所以是发送到飞书机器人). 本次实践搞 ...
- 教你轻松解决CSRF跨站请求伪造攻击
摘要:CSRF(Cross-site request forgery)跨站请求伪造,通过伪装来自受信任用户的请求来利用受信任的网站.与XSS攻击相比,CSRF攻击往往不大流行(因此对其进行防范的资源也 ...
- [AcWing 862] 三元组排序
点击查看代码 #include <iostream> #include <algorithm> using namespace std; const int N = 1e5 + ...
- 【microPython与esp8266】之一——呼吸灯与PWM
呼吸灯与pwm pwm是什么? PWM的全称是脉冲宽度调制(Pulse-width modulation),是通过将有效的电信号分散成离散形式从而来降低电信号所传递的平均功率的一种方式: 简而言之,使 ...
- Go语言学习——map
map 映射关系容器 内部使用散列表(hash)实现 map是引用类型 必须初始化才能使用 无序的基于key-value的数据结构 map定义 map的定义语法: map[KeyType]ValueT ...