Panic and Recover
Understand how to use panic and recover to handle unexpected errors in Go applications
In Go, panic
and recover
mechanisms are often used for handling unexpected errors and ensuring graceful recovery in applications. While they should not replace conventional error handling, they can be useful in certain situations.
Basic Panic and Recover Usage
Below is an example that illustrates the use of panic
and recover
in a Go program:
package main
import (
"fmt"
)
func main() {
defer func() {
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
}()
fmt.Println("Starting the program")
causePanic()
fmt.Println("This line won't be reached because of the panic")
}
func causePanic() {
panic("Unable to proceed")
}
Using Panic and Recover with HTTP Handlers
Here's how you might use panic
and recover
to ensure your HTTP server does not crash:
package main
import (
"fmt"
"net/http"
)
func main() {
http.HandleFunc("/", safeHandler(exampleHandler))
http.ListenAndServe(":8080", nil)
}
func exampleHandler(w http.ResponseWriter, r *http.Request) {
panic("Test panic")
}
func safeHandler(h http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
defer func() {
if err := recover(); err != nil {
fmt.Fprintf(w, "Recovered from panic: %v", err)
// Code continues to be executed.
}
}()
h(w, r)
}
}
Best Practices
Minimal Usage: Use
panic
andrecover
sparingly. They are intended for serious errors that are not meant to be recovered during normal operation.Log Panics: When using
recover
, always log the error. This can help identify the cause of the panic during debugging.Graceful Shutdown: Ensure resources like files, network connections, and other systems are properly closed with
defer
even when a panic occurs.Separate Error Handling: Use standard error handling practices for expected errors. Reserved
panic
for truly exceptional cases.
Common Pitfalls
Overuse of Panic: Relying on panic for normal error handling instead of returning error values or using standard error handling patterns.
Ignoring Recovery: Ensure that the recovery block responsibly handles the error and makes necessary cleanup or logging.
Non-Recovery Scenarios: Using
recover
outside of adefer
function will not catch a panic.
Performance Tips
Avoid Unnecessary Panics: Since recovering from a panic is relatively expensive, avoid unnecessary panics to maintain performance.
Limit Scope: Keep panic and recover confined to narrow scopes rather than across large sections of code to minimize the impact.
Properly Profile: Especially in high-performance systems, benchmark and profile the use of panic and recover to understand their impact on performance.
By effectively integrating and managing panic
and recover
, Go developers can improve the resilience of their applications for exceptional conditions.