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

【Golang】defer与recover的组合使用

在Go语言中,deferrecover是两个关键特性,通常结合使用以处理资源管理和异常恢复。以下是它们的核心应用场景及使用示例:


1. defer 的应用场景

defer用于延迟执行函数调用,确保在函数退出前执行特定操作。主要用途包括:

资源释放
  • 文件操作:确保文件句柄关闭。

    func readFile(filename string) error {file, err := os.Open(filename)if err != nil {return err}defer file.Close() // 确保函数返回前关闭文件// 处理文件内容...return nil
    }
    
  • 锁释放:防止死锁。

    var mu sync.Mutex
    func updateData() {mu.Lock()defer mu.Unlock() // 函数退出时自动释放锁// 修改共享数据...
    }
    
事务回滚
  • 数据库或业务逻辑中,确保操作失败时回滚。
    func transferMoney() {tx := db.Begin()defer func() {if r := recover(); r != nil { // 结合recover处理panictx.Rollback()}}()// 执行转账操作,可能触发panictx.Commit()
    }
    

2. recover 的应用场景

recover用于捕获panic,防止程序非正常终止。必须在defer函数中调用

全局异常恢复
  • 防止因未处理的panic导致程序崩溃。
    func safeCall() {defer func() {if r := recover(); r != nil {fmt.Println("Recovered from panic:", r)}}()// 可能触发panic的代码panic("unexpected error")
    }
    
保护Goroutine
  • 避免某个Goroutine的panic影响整个程序。
    func startWorker() {go func() {defer func() {if r := recover(); r != nil {log.Println("Worker panic:", r)}}()// Goroutine的业务逻辑...}()
    }
    
优雅降级
  • panic转换为错误,保持服务可用性。
    func safeHandler() (err error) {defer func() {if r := recover(); r != nil {err = fmt.Errorf("internal error: %v", r)}}()// 可能panic的代码return nil
    }
    

3. 结合使用示例

func processRequest() (err error) {// 恢复panic并转为错误defer func() {if r := recover(); r != nil {err = fmt.Errorf("panic occurred: %v", r)}}()// 资源管理示例file, err := os.Open("data.txt")if err != nil {return err}defer file.Close() // 确保文件关闭// 业务逻辑(可能触发panic)if someCondition {panic("data corruption")}return nil
}

4. 注意事项

  1. recover仅在defer中有效:非defer上下文中调用会返回nil
  2. 避免滥用recover:隐藏panic可能导致未知状态,应仅在必要时使用。
  3. 明确错误处理:优先返回错误而非依赖panic/recover,后者适用于不可恢复的异常(如程序逻辑错误)。

通过合理使用deferrecover,可以显著提升Go程序的健壮性和可维护性,尤其在资源管理和异常恢复场景中。

在Go语言中,上述代码无法正确捕获panic,原因如下:


1. 问题分析

代码示例
func safeCall() {// 直接调用recover(不在defer中)if r := recover(); r != nil {fmt.Println("Recovered from panic:", r)}// 触发新的panicpanic("unexpected error")
}// 将safeCall包裹在defer中
defer func() { safeCall() }()
关键问题
  1. recover未在defer中调用

    • safeCall中的recover直接调用,而非通过defer注册的函数。此时recover会在safeCall正常执行时立即触发,而非在panic发生后被动调用。
    • 若此时未发生panicrecover返回nil,无法捕获后续触发的panic
  2. panicdefer执行顺序

    • 当外层函数触发panic时,会先执行已注册的defer函数。
    • defer中调用safeCall,此时safeCall内的recover尝试捕获当前panic,但随后safeCall自身又触发了一个新的panic("unexpected error"),而新的panic未被任何recover处理,导致程序崩溃。

2. 正确写法

修复方案

recover放在defer函数中,并直接与可能触发panic的代码关联:

func safeCall() {// 可能触发panic的代码defer func() {// 在defer中调用recoverif r := recover(); r != nil {fmt.Println("Recovered from panic:", r)}}()panic("unexpected error")
}// 注册defer
defer safeCall()
执行逻辑
  1. 调用defer safeCall(),注册safeCall到外层函数的defer栈。
  2. 当外层函数触发panic时,执行safeCall
  3. safeCall内部的defer函数中的recover会捕获当前panic阻止其继续传播
  4. safeCall自身触发panic,该panic会被其自身的defer recover捕获。

3. 错误示例的详细解释

原代码执行流程

假设外层函数触发panic

  1. 外层函数执行panic("outer panic")
  2. 程序开始处理defer,调用defer func() { safeCall() }()
  3. safeCall执行:
    • recover()尝试捕获外层panic("outer panic"),打印恢复信息。
    • 随后触发新的panic("unexpected error")
  4. 新的panic未被任何recover处理,导致程序崩溃。
关键结论
  • recover必须通过defer注册的函数被动调用,才能捕获到panic
  • 若在普通代码中直接调用recover,只有在已发生panic且未被处理时才会生效。

4. 总结

  • 必须将recover放在defer函数中,才能确保在panic发生后被动调用。
  • 避免在恢复逻辑中触发新的panic,否则需要额外的recover处理。
  • 正确的deferrecover组合是资源管理和异常恢复的核心模式。

通过调整代码结构,确保recoverdefer中调用,即可正确捕获并处理panic

相关文章:

【Golang】defer与recover的组合使用

在Go语言中,defer和recover是两个关键特性,通常结合使用以处理资源管理和异常恢复。以下是它们的核心应用场景及使用示例: 1. defer 的应用场景 defer用于延迟执行函数调用,确保在函数退出前执行特定操作。主要用途包括&#xff…...

低代码配置式Web组态解析

低代码配置式Web组态技术通过可视化操作和预置组件库,大幅降低开发门槛,适用于工业控制、物联网监控、数据可视化等场景。以下是综合行业实践和产品特性的分析: ‌一、核心功能与优势‌ ‌可视化编辑与拖拽布局‌ 提供图形化编辑器&#xff0…...

KiLog2MaximumIncrement的由来和KiMaximumIncrementReciprocal的由来

第一部分&#xff1a;KiLog2MaximumIncrement的由来 i 1; j KeMaximumIncrement; while ((1UI64<<i) < KeMaximumIncrement) { i; } KiLog2MaximumIncrement i; 2^17131072 2^18262144 i18KiLog2MaximumIncrement 中…...

基于web的家政服务网站

内容摘要 由于互联网的使用&#xff0c;人们在管理、应用、服务等领域使用数据更加简洁、方便&#xff0c;大大提高了工作效率。互联网正逐渐融入我们的生活&#xff0c;影响和改变我们的生活。 家政服务管理系统是典型的信息管理系统&#xff08;MIS&#xff09;。其开发主要…...

mac命令行快捷键

光标移动 Ctrl A: 将光标移动到行首。Ctrl E: 将光标移动到行尾。Option 左箭头: 向左移动一个单词。Option 右箭头: 向右移动一个单词。 删除和修改 Ctrl K: 删除从光标到行尾的所有内容。Ctrl U: 删除从光标到行首的所有内容。Ctrl W: 删除光标前的一个单词。Ctrl …...

聚水潭数据集成到MySQL的最佳实践分享

聚水潭数据集成到MySQL的技术案例分享 在本次技术案例中&#xff0c;我们将探讨如何通过轻易云数据集成平台&#xff0c;将聚水潭的数据高效、可靠地集成到MySQL数据库中。具体的集成方案为“聚水潭-商品信息查询-->BI初本-商品信息表_copy”。该方案旨在实现从聚水潭获取商…...

线性代数核心概念与NumPy科学计算实战全解析

前言 学习方法&#xff1a; 思维导图&#xff0c;梳理 多记忆&#xff0c;函数名和功能&#xff0c;参数 学会应用&#xff0c;不要钻牛角尖 一、浅解线性代数 1.1标量 标量是一个只有大小没有方向的量。在数学上&#xff0c;标量通常表示为一个普通的数字&#xff0c;如‌质量…...

Spring Boot中接口数据字段为 Long 类型时,前端number精度丢失问题解决方案

Spring Boot中接口数据字段为 Long 类型时&#xff0c;前端number精度丢失问题解决方案 在Spring Boot中&#xff0c;当接口数据字段为 Long 类型时&#xff0c;返回页面的JSON中该字段通常会被序列化为数字类型。 例如&#xff0c;一个Java对象中有一个 Long 类型的属性 id …...

C#自定义曲线便器功能实现(简化版)

目录 一、曲线编辑器实现功能 二、实现方法说明 三、关键代码说明 1、绘制背景板和曲线 2、绘制坐标系面板 3、绘制曲线 四、工程下载连接 一、曲线编辑器实现功能 添加或者删除控制点&#xff0c;通过移动控制点来修改曲线形状 二、实现方法说明 1、坐标系系统&#x…...

Unity Shader编程】之复杂光照

在Unity Shader的LightMode标签中&#xff0c;除了前向渲染和延迟渲染外&#xff0c;还支持多种渲染模式设置。以下是主要分类及用途&#xff1a; 一、核心渲染路径模式 前向渲染相关 ForwardBase 用于基础光照计算&#xff0c;处理环境光、主平行光、逐顶点/SH光源及光照贴图。…...

解锁U盘属性0字节困境,重获数据生机

在数字化浪潮中&#xff0c;U盘宛如一位忠诚的“数据信使”&#xff0c;频繁穿梭于各种设备之间&#xff0c;为我们存储和传输着重要信息。然而&#xff0c;当U盘突然显示属性为0字节时&#xff0c;就如同这位信使突然“失声”&#xff0c;让我们陷入了数据丢失的恐慌之中。U盘…...

⭐算法OJ⭐二叉树的直径【树】(C++实现)Binary Tree Paths

543. Binary Tree Paths&#xff08;二叉树的直径&#xff09; Given the root of a binary tree, return the length of the diameter of the tree. The diameter of a binary tree is the length of the longest path between any two nodes in a tree. This path may or m…...

docker使用命令笔记

docker使用命令笔记 1. 安装docker2. 拉取镜像3. 镜像与容器4. 基于镜像创建容器4. 操作创建好的容器5. docker文件传输6. ubuntu的docker的一些基本环境搭建 记录docker的一些使用命令 1. 安装docker 遵循官方安装说明即可&#xff0c;windows需要下载docker desktop后在doc…...

字典树与01trie

字典树简介 当我们通过字典查一个字或单词的时候&#xff0c;我们会通过前缀或关键字的来快速定位一个字的位置&#xff0c;进行快速查找。 字典树就是类似字典中索引表的一种数据结构&#xff0c;能够帮助我们快速定位一个字符串的位置。 字典树是一种存储字符串的数据结构…...

vue - [Vue warn]: Duplicate keys detected: ‘0‘. This may cause an update error.

问题描述&#xff1a; vue项目中&#xff0c;对表单数组赋值时&#xff0c;控制台抛出警告&#xff1a; 问题代码&#xff1a; 问题分析&#xff1a; 1、Vue 要求每个虚拟 DOM 节点必须有唯一的 key。该警告信息通常出现在使用v-for循环的场景中&#xff0c;多个同级节点使用…...

mysql中show命令的使用

在 MySQL 中&#xff0c;SHOW 命令是一个非常实用的工具&#xff0c;用于查询数据库元数据&#xff08;如数据库、表、列、索引等信息&#xff09;。以下是常见的 SHOW 命令及其用法&#xff1a; 1. 显示所有数据库 SHOW DATABASES;列出服务器上的所有数据库。 2. 显示当前数据…...

各类神经网络学习:(三)RNN 循环神经网络(中集),同步多对多结构的详细解释

上一篇下一篇RNN&#xff08;上集&#xff09;RNN&#xff08;下集&#xff09; 同步多对多结构 1&#xff09;结构详解 ①图解&#xff1a; ②参数含义&#xff1a; x t x_t xt​ &#xff1a;表示每一个时刻的输入&#xff1b; o t o_t ot​ &#xff1a;表示每一个时刻的输…...

Python Web 框架 Django、Flask 和 FastAPI 对比

在探索 Python Web 框架时&#xff0c;Django、Flask 和 FastAPI 无疑是最常被提及的名字。根据我们最新的 Python 开发者调查&#xff0c;这三大框架继续稳坐后端 Web 开发的热门宝座。它们均为开源项目&#xff0c;并且与 Python 的最新版本无缝兼容。然而&#xff0c;面对不…...

Hyperlane 似乎是一个轻量级、高性能的 Rust HTTP 服务器库

关键要点 Hyperlane 是一个轻量级、高性能的 Rust HTTP 服务器库&#xff0c;适合简化网络服务开发。它支持 HTTP 请求解析、响应构建、TCP 通信&#xff0c;并提供中间件、WebSocket 和服务器发送事件&#xff08;SSE&#xff09;功能。安装通过 cargo add hyperlane 完成&am…...

【计算机网络运输层详解】

文章目录 一、前言二、运输层的功能1. 端到端通信2. 复用与分用3. 差错检测4. 流量控制5. 拥塞控制 三、运输层协议&#xff1a;TCP 和 UDP1. TCP&#xff1a;面向连接的可靠传输协议2. UDP&#xff1a;无连接的传输协议 四、端口号与进程通信1. 端口号分类2. 端口通信模型 五、…...

UR5e机器人位姿

UR5e 作为一款 6 自由度协作机器人&#xff0c;其末端执行器的位姿&#xff08;位置与姿态的组合&#xff09;控制是实现精准操作的核心。在笛卡尔坐标系中&#xff0c;位姿通常用齐次变换矩阵表示&#xff0c;包含末端的三维位置&#xff08;x, y, z&#xff09;和三维姿态&am…...

导入 Excel 规则批量修改或删除 PDF 文档内容

需要对 PDF 文档内容进行修改的时候&#xff0c;通常我们会需要借助一些专业的工具来帮我们完成。那我们如果需要修改的 PDF 文档较多的时候&#xff0c;有什么方法可以帮我们实现批量操作呢&#xff1f;今天这篇文章就给大家介绍一下当我们需要批量修改多个 PDF 文档的时候&am…...

大模型tokenizer重构流程

大模型tokenizer层再训练&#xff08;选取Qwen7B试验&#xff0c;重构token层&#xff09; 最近公司可能想训练一个蛋白质大模型&#xff0c;需要了解一下大模型tokenizer重构&#xff0c;之后可能要训练&#xff0c;这里做了一定的总结。 文章目录 1. 首先查看Qwen2.5 7B基本…...

JAVA线程安全的集合类分类

1. 传统同步集合类&#xff08;早期实现&#xff0c;性能较低&#xff09;‌ ‌Vector‌ 动态数组实现&#xff0c;所有方法通过 synchronized 同步锁保证线程安全‌。 ‌Stack‌ 继承自 Vector&#xff0c;实现后进先出&#xff08;LIFO&#xff09;堆栈&#xff0c;同步锁机…...

ISIS-1 ISIS概述

前面几章我们介绍了OSPF的基础工作原理以及怎样交互LSA形成LSDB链路状态数据库的 这一章我们来介绍另一个链路状态路由协议,ISIS路由协议 一、概述 ISIS(Intermediate System to Intermediate System,中间系统到中间系统)是由ISO(International Organization for Standardiza…...

茱元游戏TV2.9.3 | 适配多设备的经典街机游戏集合

茱元游戏TV是一款专为TV端设计的游戏软件&#xff0c;同时适配手机、投影仪和车机等多种设备。尽管其兼容性一般&#xff0c;仅支持安卓9.0以上系统&#xff0c;但它提供了丰富的经典街机游戏资源&#xff0c;非常适合8090后怀旧游玩。注意&#xff0c;游戏需先下载才能玩&…...

RTD2525BE《HDMI转EDP,DP转EDP》显示器芯片

一、产品概述 瑞昱RTD2525BE是一款专为高端显示设备设计的多接口转换芯片&#xff0c;支持HDMI 2.0与DisplayPort&#xff08;DP&#xff09;1.4双输入&#xff0c;并高效转换为嵌入式DisplayPort&#xff08;eDP&#xff09;输出。该芯片集成先进信号处理技术&#xff0c;支持…...

SvelteKit 最新中文文档教程(10)—— 部署 Cloudflare Pages 和 Cloudflare Workers

前言 Svelte&#xff0c;一个语法简洁、入门容易&#xff0c;面向未来的前端框架。 从 Svelte 诞生之初&#xff0c;就备受开发者的喜爱&#xff0c;根据统计&#xff0c;从 2019 年到 2024 年&#xff0c;连续 6 年一直是开发者最感兴趣的前端框架 No.1&#xff1a; Svelte …...

springboot使用阿里限流框架-sentinel

当前项目源码 控制台下载 启动bin中的看板服务&#xff1a;账号密码:sentinel/sentinel 官方文档地址 项目引入依赖 <!-- sentinel注解支持 --> <dependency><groupId>com.alibaba.csp</groupId><artifactId>sentinel-annotation-aspectj<…...

鸿蒙特效教程10-卡片展开/收起效果

鸿蒙特效教程10-卡片展开/收起效果 在移动应用开发中&#xff0c;卡片是一种常见且实用的UI元素&#xff0c;能够将信息以紧凑且易于理解的方式呈现给用户。 本教程将详细讲解如何在HarmonyOS中实现卡片的展开/收起效果&#xff0c;通过这个实例&#xff0c;你将掌握ArkUI中状…...