Failure to Cancel Context: Consequences and Resolution
In Go's concurrency landscape, a Context governs the lifetime and behavior of goroutines. It provides a means to propagate deadlines, cancellation signals, and other essential information. However, failing to properly cancel a Context can lead to undesired consequences.
Consider the following code snippet:
func Call(ctx context.Context, payload Payload) (Response, error) { req, err := http.NewRequest(...) // Some code that creates request from payload ctx, cancel = context.WithTimeout(ctx, time.Duration(3) * time.Second) defer cancel() return http.DefaultClient.Do(req) }
Here, a new Context is created with a timeout of 3 seconds. Without the defer cancel() statement, the function returns without explicitly cancelling the Context. This triggers a warning from go vet:
the cancel function returned by context.WithTimeout should be called, not discarded, to avoid a context leak
So, what happens if the Context is not cancelled? This omission creates a "leak" in the system. The goroutine that was responsible for handling the Context's cancellation remains active indefinitely, even though it serves no purpose. This wasted resource can result in a significant memory overhead if it occurs frequently within the program.
To avoid Context leaks, it is considered best practice to immediately defer cancel() after creating a new Context with WithCancel() or WithTimeout(). This ensures that the cancellation logic is executed as soon as the function returns, regardless of any early returns or exceptions.
In summary, neglecting to cancel a Context can lead to memory leaks by leaving goroutines running indefinitely. To prevent this, always defer the cancellation function immediately following the creation of a new Context with WithCancel() or WithTimeout().
The above is the detailed content of Why Should You Always Defer Canceling a Context in Go?. For more information, please follow other related articles on the PHP Chinese website!