你好,我是四哥。TcN28资讯网——每日最新资讯28at.com
前一篇文章从源码的角度详细介绍了 Context 的实现原理,但是还没有提到 Context 的使用场景,今天我们一起来看下:TcN28资讯网——每日最新资讯28at.com
1.请求链路传值。TcN28资讯网——每日最新资讯28at.com
传值使用方式如下:TcN28资讯网——每日最新资讯28at.com
func func1(ctx context.Context) {TcN28资讯网——每日最新资讯28at.com
ctx = context.WithValue(ctx, "k1", "v1")TcN28资讯网——每日最新资讯28at.com
func2(ctx)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func func2(ctx context.Context) {TcN28资讯网——每日最新资讯28at.com
fmt.Println("func2:",ctx.Value("k1").(string))TcN28资讯网——每日最新资讯28at.com
ctx = context.WithValue(ctx, "k2", "v2")TcN28资讯网——每日最新资讯28at.com
func3(ctx)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func func3(ctx context.Context) {TcN28资讯网——每日最新资讯28at.com
fmt.Println("func3:",ctx.Value("k1").(string))TcN28资讯网——每日最新资讯28at.com
fmt.Println("func3:",ctx.Value("k2").(string))TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
ctx := context.Background()TcN28资讯网——每日最新资讯28at.com
func1(ctx)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
我们在 func1() 通过函数 WithValue() 设置了一个键值对 k1-v1,在 func2() 可以获取到 func1() 设置的键值对,如果调用 func3() 时把这个 ctx 继续传入的话,在 func3() 中依然还是可以获取到 k1-v1。TcN28资讯网——每日最新资讯28at.com
但是在 func1() 中获取不到 func2() 设置的键值对 k2-v2,因为 context 只能自上而下携带值,这点需要注意。TcN28资讯网——每日最新资讯28at.com
2.取消耗时操作,及时释放资源。TcN28资讯网——每日最新资讯28at.com
使用 channel + select 的机制:TcN28资讯网——每日最新资讯28at.com
func func1() error {TcN28资讯网——每日最新资讯28at.com
respC := make(chan int) // 起消息通知作用TcN28资讯网——每日最新资讯28at.com
// 处理逻辑TcN28资讯网——每日最新资讯28at.com
go func() {TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second * 3) // 模拟处理业务逻辑TcN28资讯网——每日最新资讯28at.com
respC close(respC)TcN28资讯网——每日最新资讯28at.com
}()TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
// 判断是否超时TcN28资讯网——每日最新资讯28at.com
select {TcN28资讯网——每日最新资讯28at.com
case r := <-respC:TcN28资讯网——每日最新资讯28at.com
fmt.Printf("Resp: %d ", r)TcN28资讯网——每日最新资讯28at.com
return nilTcN28资讯网——每日最新资讯28at.com
case <-time.After(time.Second * 2): // 超过设置的时间就报错TcN28资讯网——每日最新资讯28at.com
fmt.Println("catch timeout")TcN28资讯网——每日最新资讯28at.com
return errors.New("timeout")TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
err := func1()TcN28资讯网——每日最新资讯28at.com
fmt.Printf("func1 error: %v ", err)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
上面的方式平时也会用到,通过 context 怎么实现呢?TcN28资讯网——每日最新资讯28at.com
下面来看下如何使用 context 进行主动取消、超时取消。TcN28资讯网——每日最新资讯28at.com
主动取消:TcN28资讯网——每日最新资讯28at.com
func func1(ctx context.Context, wg *sync.WaitGroup) error {TcN28资讯网——每日最新资讯28at.com
defer wg.Done()TcN28资讯网——每日最新资讯28at.com
respC := make(chan int)TcN28资讯网——每日最新资讯28at.com
go func() {TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second * 5) // 模拟业务逻辑处理TcN28资讯网——每日最新资讯28at.com
respC }()TcN28资讯网——每日最新资讯28at.com
// 取消机制TcN28资讯网——每日最新资讯28at.com
select {TcN28资讯网——每日最新资讯28at.com
case <-ctx.Done():TcN28资讯网——每日最新资讯28at.com
fmt.Println("cancel")TcN28资讯网——每日最新资讯28at.com
return errors.New("cancel")TcN28资讯网——每日最新资讯28at.com
case r := <-respC:TcN28资讯网——每日最新资讯28at.com
fmt.Println(r)TcN28资讯网——每日最新资讯28at.com
return nilTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
wg := &sync.WaitGroup{}TcN28资讯网——每日最新资讯28at.com
ctx, cancel := context.WithCancel(context.Background())TcN28资讯网——每日最新资讯28at.com
wg.Add(1)TcN28资讯网——每日最新资讯28at.com
go func1(ctx, wg)TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second * 2)TcN28资讯网——每日最新资讯28at.com
cancel() // 主动取消TcN28资讯网——每日最新资讯28at.com
wg.Wait() // 等待 goroutine 退出TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
超时取消:TcN28资讯网——每日最新资讯28at.com
func func1(ctx context.Context) {TcN28资讯网——每日最新资讯28at.com
resp := make(chan int)TcN28资讯网——每日最新资讯28at.com
go func() {TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second * 5) // 模拟处理逻辑TcN28资讯网——每日最新资讯28at.com
resp }()TcN28资讯网——每日最新资讯28at.com
// 超时机制TcN28资讯网——每日最新资讯28at.com
select {TcN28资讯网——每日最新资讯28at.com
case <-ctx.Done():TcN28资讯网——每日最新资讯28at.com
fmt.Println("ctx timeout")TcN28资讯网——每日最新资讯28at.com
fmt.Println(ctx.Err())TcN28资讯网——每日最新资讯28at.com
case <-resp:TcN28资讯网——每日最新资讯28at.com
fmt.Println("done")TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
returnTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
ctx, cancel := context.WithTimeout(context.Background(), time.Second*2)TcN28资讯网——每日最新资讯28at.com
defer cancel()TcN28资讯网——每日最新资讯28at.com
func1(ctx)TcN28资讯网——每日最新资讯28at.com
}3.防止 goroutine 泄露。TcN28资讯网——每日最新资讯28at.com
引自【深度解密 Go 语言之 context[1]】TcN28资讯网——每日最新资讯28at.com
func gen() ch := make(chan int)TcN28资讯网——每日最新资讯28at.com
go func() {TcN28资讯网——每日最新资讯28at.com
var n intTcN28资讯网——每日最新资讯28at.com
for {TcN28资讯网——每日最新资讯28at.com
ch n++TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}()TcN28资讯网——每日最新资讯28at.com
return chTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
这是一个可以生成无限整数的协程,但如果我只需要它产生的前 5 个数,那么就会发生 goroutine 泄漏:TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
for n := range gen() {TcN28资讯网——每日最新资讯28at.com
fmt.Println(n)TcN28资讯网——每日最新资讯28at.com
if n == 5 {TcN28资讯网——每日最新资讯28at.com
breakTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
// ……TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
当 n == 5 的时候,直接 break 掉。那么 gen 函数的协程就会执行无限循环,永远不会停下来。发生了 goroutine 泄漏。TcN28资讯网——每日最新资讯28at.com
用 context 改进这个例子:TcN28资讯网——每日最新资讯28at.com
func gen(ctx context.Context) ch := make(chan int)TcN28资讯网——每日最新资讯28at.com
go func() {TcN28资讯网——每日最新资讯28at.com
var n intTcN28资讯网——每日最新资讯28at.com
for {TcN28资讯网——每日最新资讯28at.com
select {TcN28资讯网——每日最新资讯28at.com
case <-ctx.Done():TcN28资讯网——每日最新资讯28at.com
returnTcN28资讯网——每日最新资讯28at.com
case ch n++TcN28资讯网——每日最新资讯28at.com
time.Sleep(time.Second)TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}()TcN28资讯网——每日最新资讯28at.com
return chTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
func main() {TcN28资讯网——每日最新资讯28at.com
ctx, cancel := context.WithCancel(context.Background())TcN28资讯网——每日最新资讯28at.com
defer cancel() // 避免其他地方忘记 cancel,且重复调用不影响TcN28资讯网——每日最新资讯28at.com
TcN28资讯网——每日最新资讯28at.com
for n := range gen(ctx) {TcN28资讯网——每日最新资讯28at.com
fmt.Println(n)TcN28资讯网——每日最新资讯28at.com
if n == 5 {TcN28资讯网——每日最新资讯28at.com
cancel()TcN28资讯网——每日最新资讯28at.com
breakTcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
// ……TcN28资讯网——每日最新资讯28at.com
}TcN28资讯网——每日最新资讯28at.com
增加一个 context,在 break 前调用 cancel 函数,取消 goroutine。gen 函数在接收到取消信号后,直接退出,系统回收资源。TcN28资讯网——每日最新资讯28at.com
总结TcN28资讯网——每日最新资讯28at.com
这篇文章列出的几个例子是 context 最基本的使用场景,其他框架、第三包基本上都是从这几种用法扩展的,所以非常有必要掌握基础用法。TcN28资讯网——每日最新资讯28at.com
另外希望这篇文章能给你带来帮助,如果文中有理解错误之处或者你还想到其他用法,可以在留言区留言,一定回复!抱团学习不孤单!TcN28资讯网——每日最新资讯28at.com
参考资料TcN28资讯网——每日最新资讯28at.com
[1]深度解密Go语言之context: https://qcrao.com/2019/06/12/dive-into-go-context/TcN28资讯网——每日最新资讯28at.com
本文链接://www.dmpip.com//www.dmpip.com/showinfo-119-2167-0.html细数 Context 使用场景
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com
上一篇: 云计算开发:Python3-isdecimal()方法详解
下一篇: 数据分析八大模型:漏斗模型