Golang Context Cancel Và Cách Sử Dụng

Context là một phần rất hay trong golang mà lập trình viên về golang cần biết, hôm nay mình sẽ nói về Cancel trong context. Context WithCancel Cancel là copy ra một context mới từ context cha trước đó kèm theo New Channel Done, nó được gửi thông tin đến rất nhiều tiến trình cũng

Context là một phần rất hay trong golang mà lập trình viên về golang cần biết, hôm nay mình sẽ nói về Cancel trong context.


Context WithCancel

Cancel là copy ra một context mới từ context cha trước đó kèm theo New Channel Done, nó được gửi thông tin đến rất nhiều tiến trình cũng như operation khác nhau. Cancel là một tính hiệu (signals) để huỷ tiến trình đang chạy ở một nơi khác(một function khác).

Tại sao chúng ta cần sử dụng Cancel:
Khi user gửi request từ client đến server nếu không có gì xảy ra khác thường thì sẽ là thế này:

Trong trường hợp nếu user đang gửi request mà cancel thì điều gì xảy ra:

nếu đã cancel mà vẫn trả về data thì là việc dư thừa không cần thiết, cũng như tốn resources để xử lý.

Hướng Giải Quyết

Để giải quyết vấn đề nầy thì cancel ra đời, mục đích là để stop ngay lập tức các operation khác, không tốn tài nguyên sử lý.

Code Demo

Ví Dụ 1:

package main

import (
	"context"
	"fmt"
)

func main() {

	taskfn := func(ctx context.Context) <-chan int {
		dst := make(chan int)
		n := 1
		go func() {
			for {
				select {
				case <-ctx.Done():
					fmt.Println("exe cannel")
					return // returning not to leak the goroutine
				case dst <- n:
					n++
				}
			}
		}()
		return dst
	}

	ctx, cancel := context.WithCancel(context.Background())
	defer cancel() // cancel when we are finished consuming integers
	for n := range taskfn(ctx) {
		fmt.Println(n)
		if n == 5 {
			break
		}
	}

}

1
2
3
4
5

Ví Dụ 2:

package main

import (
	"context"
	"errors"
	"fmt"
	"time"
)


func func1(ctx context.Context) error {
	time.Sleep(100 * time.Millisecond)
	return errors.New("failed")
}

func func2(ctx context.Context) {
	fmt.Println("exe operation2")
	select {
	case <-time.After(50 * time.Millisecond):
		fmt.Println("done")
	case <-ctx.Done():
		fmt.Println("halted operation2")
	}
}

func main() {
	ctx, cancel := context.WithCancel(context.Background())
	go func() {
		err := func1(ctx)
		fmt.Println("operation1 err:", err)
		if err != nil {
			cancel()
		}
	}()

	func2(ctx)
}

=> ở phần ví dụ này thì cancel() sẽ không được execute vì select case nó chỉ sleep 50 Millisecond.
kêt quả

exe operation2

done

tiếp theo đổi từ 50 lên 500 millisecond thì function cancel() sẽ được exec ;

case <-time.After(50 * time.Millisecond):

=> kết quả thế này:

exe operation2

operation1 err: failed

halted operation2

Cảm ơn nhé các bạn.

Source:
https://pkg.go.dev/context#WithCancel

Nguồn: viblo.asia

Bài viết liên quan

Thay đổi Package Name của Android Studio dể dàng với plugin APR

Nếu bạn đang gặp khó khăn hoặc bế tắc trong việc thay đổi package name trong And

Lỗi không Update Meta_Value Khi thay thế hình ảnh cũ bằng hình ảnh mới trong WordPress

Mã dưới đây hoạt động tốt có 1 lỗi không update được postmeta ” meta_key=

Bài 1 – React Native DevOps các khái niệm và các cài đặt căn bản

Hướng dẫn setup jenkins agent để bắt đầu build mobile bằng jenkins cho devloper an t

Chuyển đổi từ monolith sang microservices qua ví dụ

1. Why microservices? Microservices là kiến trúc hệ thống phần mềm hướng dịch vụ,