当前位置: 首页 > java >正文

defer学习指南

一、源头:

早期管理资源(如数据库连接、锁、文件句柄、网络连接)和状态清理异常麻烦。
必须在每个可能的返回点(return、err、panic)手动重复清理代码,极易遗漏且打断主要逻辑思路!

像Java语言虽然用了Try-Catch,但缺点是逻辑不清晰、臃肿、不容易判断错误出在什么地方。

作为新生代的Go,及众多语言之精华,推出了defer处理机制。
尤其是在Go1.14版本时,性能开销接近零,更无后顾之忧。

二、定义:

1、语法:

defer functionCall(..arguments..)

defer后面直接跟一个函数调用(可以是命名函数/匿名函数/方法...)

2、定义:

当函数执行到defer语句时(注册时),他会立即求值此时该函数调用的参数并将此次函数调用(包括已求值的参数)放到一个延迟调用表中。这个调用函数与goroutine关联,采用LIFO(后进先出的方式调用)。切记这个延迟调用表不会立即执行,而是会等到(函数真正结束之--函数(return或panic)之)在调用。

3、特性:

1、延迟执行:运行到defer时,只是将求值后的参数与调用的函数一并打包到延迟调用表中,需等到函数体结束之后在执行

2、LIFO方式执行:后进先出的方式执行

3、参数求值时机:defer 语句中的函数参数的值,是在执行到defer语句时(即注册时)就确定并保存下来的,而不是在延迟函数实际执行时才求值。

4、作用域:自各函数体

三、应用:

1、临近释放:(逻辑清晰)

mu.Lock()
defer mu.Unlock() // 好习惯!确保解锁
// ... 操作共享数据 ...

2、panic补获:(防止程序崩溃)

特殊情况,根据源码分析---协程中出现panic,若不能再该协程中捕获,则会导致整个程序崩溃。

func test(){defer func(){if r := recover(); r!=nil{fmt.Println(r);}}()panic(1);
}

3、循环函数释放:(利用完资源后,及时释放资源)

// 正确做法:将文件处理封装到函数,defer 在每次循环的匿名函数结束时执行
func outerFunc() {for _, filename := range filenames {func() { // 匿名函数f, err := os.Open(filename)if err != nil {log.Println(err)return // 退出匿名函数}defer f.Close() // 延迟到当前匿名函数结束时执行 (即本次循环结束)// ... 处理 f ...}() // 立即调用匿名函数}
}

4、查看执行顺序:

代码右上角,有个运行小按钮,点击运行查看。

package mainimport ("fmt""log"
)func g(i int) {if i > 1 {fmt.Println("Panicking!")panic(1)}defer fmt.Println("Defer in g", i)fmt.Println("Printing in g", i)g(i + 1)
}func f() {defer func() {if r := recover(); r != nil {log.Println("Recovered in f", r)}}()fmt.Println("Calling g.")g(0)fmt.Println("Returned normally from f.")
}func main() {f()
}

四、底层:

这个是我扒出来的底层源码,重点了解heap、link这俩。

type _defer struct {heap      bool  	//表示是分配在堆上还是栈上。rangefunc bool    	// true for rangefunc listsp        uintptr 	// 栈指针pc        uintptr 	// 程序计数器fn        func()  	// 表示需要被延迟执行的函数。link      *_defer 	// 指向下一个 _defer 结构体的指针。// If rangefunc is true, *head is the head of the atomic linked list// during a range-over-func execution.head *atomic.Pointer[_defer]
}
  • defer 语句注册时,会创建一个 _defer 结构体实例。

  • 多个 defer 通过 link 字段形成一个单链表(LIFO 栈),挂载到当前 goroutine 的结构上(g._defer)。新的 defer 总是插入链表头部。

主要有两大种分配方式。

1、堆栈分配

区别:分配位置的不同

获取到runtime_defer结构体,它都会被追加到所在 Goroutine _defer 链表的最前面。

2、开放编码

不建额外结构,直接把 defer 代码塞到函数退出前,用位掩码控制执行,开销几乎和普通调用一样。

3、选择:

首先考虑开放编码(已经优化到:实际消耗跟调用普通函数差不多的地步),后栈分配保底堆分配

以下是整理的Go版本迭代全史,有兴趣的可以一看,挺有趣的。


📅 Go 版本迭代全史(2009–2025)

⭐ 早期阶段
版本发布时间核心特性
初始开源2009-11-10正式开源,获得 TIOBE 年度语言称号
Go r562011-03-16首个稳定版本
Go 1.02012-03-28首个正式版本,承诺向后兼容性;引入 go tool pprofgo vet

🔄 每半年发布周期(2013 年起)
版本发布时间核心特性
Go 1.12013-05-13重写调度器(支持 Work-Stealing 算法);引入竞态检测器
Go 1.22013-12-01支持全切片表达式;go test 支持覆盖率统计
Go 1.32014-06-18栈模型改为连续栈;引入 sync.Pool
Go 1.42014-12-10支持 Android;运行时从 C 改为 Go;移除 src/pkg 层级
Go 1.52015-08-19自举(移除 C 代码);优化 GC(延迟降至 30ms);引入 vendor 机制
Go 1.62016-02-17默认支持 HTTP/2;GC 延迟进一步降低
Go 1.72016-08-15引入 context 包;SSA 后端优化(性能提升 5–35%)
Go 1.82017-02-17GC 延迟降至亚毫秒级;defer 性能提升 50%
Go 1.92017-08-24引入类型别名;新增并发安全的 sync.Map
Go 1.102018-02-16构建缓存(Build Cache)默认开启
Go 1.112018-08-25引入 Go Modules;支持 WebAssembly
Go 1.122019-03-01优化 TLS 1.3 支持;改进模块机制
Go 1.132019-09-03支持二进制/八进制字面量;错误处理增强(errors.Is/As/Unwrap
Go 1.142020-02-25接口允许方法集重叠;Goroutine 支持异步抢占调度;defer 性能接近零开销
Go 1.152020-08-11优化链接器;改进内联策略
Go 1.162021-02支持静态文件嵌入;默认启用 Go Modules

🚀 重大革新阶段(2022–2025)
版本发布时间核心特性
Go 1.182022-03-15引入泛型;支持模糊测试(Fuzzing);工作区模式(Multi-Module Workspaces)
Go 1.232025 年初引入迭代器(seq/seq2);gopls 现代化工具链;go get 管理工具链
Go 1.242025 年中标准库支持 strings/slices/maps 迭代器;增强 WebAssembly 安全性与性能
Go 1.252025-08 (预计)移除核心类型(Core Types);简化泛型规范;优化错误提示

太多了不好记,有兴趣查看时,可以重点看

1、初始开源-2009-11-10,go降生到了这个世界上

2、Go1.5 :2015,go开始用母语了,实现自举(Go 编译 Go),GC 延迟从 300ms 降至 30ms,奠定现代 Go 基础

3、Go1.11:2018,引入了Go Module解决了依赖问题,让现在的我都收益不止--今年2025

4、Go1.18:2022,泛型、模糊测试、工作区多模块,等均进行了新功能的填充你与优化。这个咱暂接触不够多,需要优化

.....

 

http://www.xdnf.cn/news/14989.html

相关文章:

  • 黑搜小知识 | DNS域名解析过程是什么样的?
  • 【Modern C++ Part8】Prefer-nullptr-to-0-and-NULL
  • 深度学习12(卷积神经网络)
  • Token 和 Embedding的关系
  • 模型内部进行特征提取时,除了“减法”之外,还有哪些技术
  • SpringCloud系列 - xxl-job 分布式任务调度 (七)
  • Using Spring for Apache Pulsar:Publishing and Consuming Partitioned Topics
  • swiglu 激活函数学习笔记
  • Rust与Cypress应用
  • 技术支持丨解决 ServBay 在 Windows 启动时反复提示安装 .NET 的问题
  • Flask3.1打造极简CMS系统
  • leetcode11.盛最多水的容器
  • 微信小程序91~100
  • STM32-待机唤醒实验
  • 搭建一款结合传统黄历功能的日历小程序
  • S7-200 SMART :通过以太网下载程序详细步骤
  • ServBay Windows 1.2.0 更新!新增 PHP 设置与 Ollama 支持
  • Docker 高级管理 -- 容器通信技术与数据持久化
  • 人工智能-基础篇-27-模型上下文协议--MCP到底怎么理解?对比HTTP的区别?
  • 如何卸载本机的node.js
  • 【视频观看系统】- 需求分析
  • 沃丰科技海外客服系统综合解决方案
  • 【DB2】load报错SQL3501W、SQL3109N、SQL2036N
  • 持续更新!国内免费使用 claude code 方案
  • LLaMA-Omni 深度解析:打开通往无缝人机语音交互的大门
  • C++学习笔记三
  • 使用 Docker Compose 简化 INFINI Console 与 Easysearch 环境搭建
  • 跨部门协作难以对齐项目进度,如何促进协同
  • 【动手学深度学习】4.10 实战Kaggle比赛:预测房价
  • S7-1500——(一)从入门到精通1、基于TIA 博途解析PLC程序结构(一)