Go panic 用法详细介绍
什么是 Panic?
在 Go 程序中处理异常情况的惯用方法是使用错误处理。 对于程序中出现的大多数异常情况,错误就足够了。
但也有一些情况,程序在出现异常情况后无法继续执行。 在这种情况下,我们使用 panic 提前终止程序。 当一个函数遇到panic 时,它的执行会停止,任何延迟的函数都会被执行,然后控制权返回给它的调用者。 这个过程一直持续到当前 goroutine 的所有函数都已经返回,此时程序会打印 panic 消息,然后是堆栈跟踪,最后终止。 通过示例程序时,我们会对这个概念有更加清晰的认识。
可以使用我们将在本章节后面讨论的 Recover
来重新控制 Panic的程序。
panic 和 recover 可以被认为类似于 Java 等其他语言中的 try-catch-finally
语法。
什么时候使用 Panic?
一个重要的因素是我们应该尽量避免 Panic 和 Recover,并在可能的情况下使用错误。 只有在程序无法继续执行的情况下才应使用 panic 和 recover 机制。
下面是两个有效的使用 panic 的用例
1. 发生不可恢复的错误导致程序无法继续执行
一个示例是无法绑定到所需端口的 Web 服务器。 在这种情况下,使用 Panic 是合理的,因为如果端口绑定本身失败,则无事可做。
2. 程序员导致的错误
假设我们有一个接受指针作为参数的方法,并且有人使用 nil 参数调用此方法。 在这种情况下,我们可能会使用 panic,因为使用 nil 参数调用那些需要有效指针才能调用的方法是程序员犯的错误。
Panic 示例
下面是内置的 Panic 函数的声明
func panic(interface{})
传递给 panic 函数的参数将在程序终止时打印出来。
我们将从一个例子开始,展示了 panic 是如何工作的。
package main
import (
"fmt"
)
func fullName(firstName *string, lastName *string) {
if firstName == nil {
panic("runtime error: first name cannot be nil")
}
if lastName == nil {
panic("runtime error: last name cannot be nil")
}
fmt.Printf("%s %s\n", *firstName, *lastName)
fmt.Println("returned normally from fullName")
}
func main() {
firstName := "Elon"
fullName(&firstName, nil)
fmt.Println("returned normally from main")
}
以上是一个简单的程序,它打印一个人的全名。 fullName 函数打印一个人的全名。 这个函数检查第一个名字和最后一个名字的指针在是否为nil。 如果它是 nil 函数调用 panic 并带有相应的消息。 程序终止时将打印此消息。
运行以上程序将打印如下错误,我们也可以点击上面的运行示例按钮直接在线运行上述程序查看结果
panic: runtime error: last name cannot be nil
goroutine 1 [running]:
main.fullName(0xc000054f58, 0x0)
/tmp/main.go:12 +0x19d
main.main()
/tmp/main.go:20 +0x4d
exit status 2
让我们分析这个输出来了解 Panic 是如何工作的,以及当程序遇到 Panic 时堆栈跟踪是如何打印信息的。
我们将 “Elon” 分配给 firstName。 我们调用 fullName
函数,lastName 为 nil。 因此,条件会被满足,程序会崩溃。 当遇到 Panic 时,程序执行终止,传递给 Panic 函数的参数被打印出来,然后是堆栈跟踪。 由于程序在 panic 函数调用之后终止。后续的代码将不会被执行。
这个程序首先打印传递给 panic 函数的消息
panic: runtime error: last name cannot be nil
然后打印堆栈信息。程序在fullName函数中panic,因为lastName为nil。
goroutine 1 [running]:
main.fullName(0xc000054f58, 0x0)
/tmp/main.go:12 +0x19d
然后将打印堆栈中的下一个项目。 在我们的例子中,第20行调用 fullName 的地方是堆栈跟踪中的下一项。 因此它接下来被打印。
main.main()
/tmp/main.go:20 +0x4d
exit status 2
现在我们已经到达了导致Panic的最上层的函数调用,上面没有更高的级别,因此没有更多的东西要打印。
下面我们再看一个示例
运行时发生的错误也可能导致 Panic ,例如尝试访问切片中不存在的索引。
让我们编写一个示例,该示例由于越界切片访问而导致Panic。
package main
import (
"fmt"
)
func slicePanic() {
n := []int{5, 7, 4}
fmt.Println(n[4])
fmt.Println("normally returned from a")
}
func main() {
slicePanic()
fmt.Println("normally returned from main")
}
在上面的程序中,我们试图访问 n[4]
,这是切片中的无效索引。 该程序运行显示如下结果。
panic: runtime error: index out of range [4] with length 3
goroutine 1 [running]:
main.slicePanic()
/tmp/main.go:9 +0x1d
main.main()
/tmp/main.go:13 +0x25
exit status 2
下一篇我们介绍 Go panic 和 defer 组合使用
相关文章
在 Golang 中使用 If-Else 和 Switch Loop Inside HTML 模板
发布时间:2023/04/27 浏览次数:65 分类:Go
-
本篇文章介绍了在 Golang 的 HTML 模板中使用 if-else 和 switch 循环。因此,只要输出是 HTML,就应该始终使用 HTML 模板包而不是文本模板。
Golang 中的零值 Nil
发布时间:2023/04/27 浏览次数:166 分类:Go
-
本篇文章介绍 nil 在 Golang 中的含义,nil 是 Go 编程语言中的零值,是众所周知且重要的预定义标识符。
Golang 中的 Lambda 表达式
发布时间:2023/04/27 浏览次数:93 分类:Go
-
本篇文章介绍如何在 Golang 中创建 lambda 表达式。Lambda 表达式似乎不存在于 Golang 中。 函数文字、lambda 函数或闭包是匿名函数的另一个名称。
在 Go 中捕获 Panics
发布时间:2023/04/27 浏览次数:66 分类:Go
-
像错误一样,Panic 发生在运行时。 换句话说,当您的 Go 程序中出现意外情况导致执行终止时,就会发生 Panics。让我们看一些例子来捕捉 Golang 中的Panics。
在 Go 中使用断言
发布时间:2023/04/27 浏览次数:181 分类:Go
-
本篇文章介绍了 assert 在 GoLang 中的使用。在 Go 语言中使用断言:GoLang 不提供对断言的任何内置支持,但我们可以使用来自 Testify API 的广泛使用的第三方包断言。
Go 中的随机数生成
发布时间:2023/04/27 浏览次数:114 分类:Go
-
本篇文章介绍如何在 Go 语言中使用随机数生成功能。Go 中的随机数生成 Go 语言为随机数生成功能提供内置支持。 内置包 math 有方法 rand(),用于随机数生成。