程序员求职经验分享与学习资料整理平台

网站首页 > 文章精选 正文

Go语言零到一:panic 和 recover 用法

balukai 2025-07-14 14:50:05 文章精选 2 ℃

引言

在 Go 语言中,panicrecover 是用于处理运行时错误的特殊机制。panic 用于报告不可恢复的错误,而 recover 用于从 panic 中恢复,防止程序崩溃。

1. panic 机制

  • 触发 panic
    • 当程序遇到严重错误,无法正常执行时,可以调用 panic 函数来抛出一个 panic。
    • panic 接受一个 interface{} 类型的参数,通常是一个字符串或错误对象。
    • 示例:
func divide(x, y int) int { 
 if y == 0 { 
 panic("division by zero") 
 } 
 return x / y 
}
  • panic 的影响
    • 一旦发生 panic,当前 goroutine 的所有函数调用都会立即停止执行。
    • 所有 defer 语句将被依次执行,直到 panic 被 recover 捕获或 goroutine 终止。

2. recover 机制

  • 捕获 panic
    • 使用 deferrecover 函数可以在 defer 函数中捕获 panic。
    • recover 函数只能在 defer 函数中调用,否则会引发编译错误。
    • 示例:
func safeDivide(x, y int) int {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Recovered from panic:", r)
		}
	}()
	return divide(x, y)
}
  • 恢复执行
    • 如果 recover 成功捕获了 panic,它会返回 panic 的值,goroutine 可以继续执行。
    • 如果 recover 没有捕获到 panic,goroutine 将终止,panic 将传递给调用者。

3. 使用场景

  • 资源清理
    • 在 panic 发生时,使用 defer 和 recover 来确保资源被正确释放。
    • 示例:
func processFile(filename string) {
	file, err := os.Open(filename)
	if err != nil {
		panic(err)
	}
	defer func() {
		if err := recover(); err != nil {
			fmt.Println("Error processing file:", err)
			file.Close()
		}
	}()
	// 文件处理逻辑
	file.Close()
}
  • 错误恢复
    • 在可能产生 panic 的函数中使用 defer 和 recover 来尝试恢复错误,避免程序崩溃。
    • 示例:
func handleRequest(w http.ResponseWriter, r *http.Request) {
	defer func() {
		if err := recover(); err != nil {
			http.Error(w, "Internal Server Error", http.StatusInternalServerError)
		}
	}()
	// 请求处理逻辑
}

4.编码

package main

import (
	"fmt"
	"os"
)

func divide(x, y int) int {
	if y == 0 {
		panic("division by zero")
	}
	return x / y
}

func safeDivide(x, y int) int {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("Recovered from panic:", r)
		}
	}()
	return divide(x, y)
}

func main() {
	// 正常情况
	fmt.Println(safeDivide(10, 2))

	// 触发 panic
	fmt.Println(safeDivide(10, 0))

	// 打开文件并处理潜在的 panic
	file, err := os.Open("nonexistentfile.txt")
	if err != nil {
		panic(err)
	}
	defer func() {
		if err := recover(); err != nil {
			fmt.Println("Error opening file:", err)
			file.Close()
		}
	}()
	// 文件处理逻辑
	file.Close()
}

Tags:

最近发表
标签列表