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.
Nguồn: viblo.asia