golang的协程上下文的具体使用

作者:zhijie 时间:2024-02-01 00:41:02 

go协程上下文context

golang的context 主要用来在 goroutine 之间传递上下文信息,包括:取消信号、超时时间、截止时间、k-v 等

context是golang1.17版本之后才出的特性

上下文解决的问题

  • 协程间的通信

例如web应用中,每一个请求都由一个协程去处理。当然处理处理请求的这个协程,一般我们还会起一些其他的协程,用来处理其他的业务,比如操作数据库,生份验证、文件读写等。这些协程是独立的,我们在当前的协程中无法感知到其他的协程执行的情况怎么样了。实用通道channel可以实现通讯功能

contextcontext.WithValue()本质上也是通过channel来实现通讯

  • 子协程的超时处理

同样例如web应用当中,我们主进程是一直常驻内存的。每一个请求都由一个协程去处理,在处理业务的过程中可能会起另外的协程去处理其他的业务,当子协程出现了异常或者阻塞了,无法向上一级的协程反馈信息,主协程接受不到反馈也会阻塞。上下文可以很好的解决这个问题,context可以实现子协程或子孙协程的超时退出或定时退出

上下文的几个方法和用法

context.WithCancel(context.Context)

WithCancel()方法传入一个上下文空实例,直接用context.Background()即可,返回一个上下文和一个取消函数。调用cancal()会向其他协程传递信号,收到信号后子协程就可以做关闭或其他处理

package main

import (
   "context"
   "fmt"
   "time"
)
func worker(ctx context.Context) {
   for {
       select {
       case <-ctx.Done():
           return
       default:
       }
       fmt.Println("worker...")
       time.Sleep(time.Second * 1)
   }
}
func main() {
   ctx, cancal := context.WithCancel(context.Background())
   go worker(ctx)
   time.Sleep(time.Second * 5)
   cancal()
   fmt.Println("over ...")

}

context.WithTimeout(context.Context,timeout)

定义一个会超时的上下文,实例化后倒计时就开始,到时间会自动调用cancel()函数通知子协程,也可以手动调用cancel()通知。如果子协程中还有子协程,继续使用这个上下文,当主协程发出取消信号时每一个使用了这个上下文的都会收到通知

package main

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

func worker(ctx context.Context) {
   for {
       select {
       case <-ctx.Done():
           return
       default:
       }
       fmt.Println("worker...")
       time.Sleep(time.Second * 1)
   }
}

func main() {
   ctx, cancal := context.WithTimeout(context.Background(), time.Second*2)
   go worker(ctx)
   time.Sleep(time.Second * 5)
   cancal()
   fmt.Println("over ...")

}

context.WithDeadline(context.Context,(绝对时间)timeout)

定义一个会超时的上下文,与Timeout不同在于,传入的时间是一个绝对时间。到了指定的时间会自动调用cancel()函数通知子协程,也可以手动调用cancel()通知。如果子协程中还有子协程,继续使用这个上下文,当主协程发出取消信号时每一个使用了这个上下文的都会收到通知

package main

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

func worker(ctx context.Context) {
   for {
       select {
       case <-ctx.Done():
           return
       default:
       }
       fmt.Println("worker...")
       time.Sleep(time.Second * 1)
   }
}

func main() {
   ctx, cancal := context.WithDeadline(context.Background(), time.Now().Add(3 * time.Second))
   go worker(ctx)
   time.Sleep(time.Second * 5)
   cancal()
   fmt.Println("over ...")

}

协程间的上下文通讯:context.WithValue()

先看一下这段代码

package main

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

type CTXKEY string

func worker(ctx context.Context) {
   // 在子协程中获取上下文信息
   num, ok := ctx.Value(CTXKEY("num")).(string)
   if !ok {
       fmt.Println("invalid trace code")
   }
   fmt.Println(num)
   for {
       select {
       case <-ctx.Done():
           return
       default:
       }
       fmt.Println("worker...")
       time.Sleep(time.Second * 1)
   }
}

func main() {
   ctx, cancal := context.WithDeadline(context.Background(), time.Now().Add(3*time.Second))
   // 利用上下文传一个 num = 1234567
   // 实例化一个上下文
   ctx = context.WithValue(ctx, CTXKEY("num"), "1234567")
   go worker(ctx)
   time.Sleep(time.Second * 5)
   cancal()
   fmt.Println("over ...")
}

通过上下文实现协程间的通信,如果项目大,为了避免变量的污染,原则上:上下文通信所用的key需要自定义一个类型

type traceCode string;context.WithValue(context.Context,key,value)

来源:https://juejin.cn/post/7084613288104820773

标签:golang,协程上下文,context
0
投稿

猜你喜欢

  • 自动定时备份sqlserver数据库的方法

    2024-01-13 20:45:14
  • 数据库基础:MySQL 添加用户的两种方法

    2009-05-07 14:26:00
  • 何时将数据装载到Application 或 Session 对象中去?

    2009-12-03 20:17:00
  • Python动态导入模块和反射机制详解

    2023-07-16 14:02:10
  • Python 程序报错崩溃后如何倒回到崩溃的位置(推荐)

    2021-01-08 16:14:34
  • 谈谈XHTML中CDATA

    2007-09-17 12:45:00
  • Python实现的读写json文件功能示例

    2023-11-29 21:42:41
  • vue项目依赖升级报错处理方式

    2024-05-10 14:20:52
  • vue引入jq插件的实例讲解

    2024-04-30 10:23:11
  • mysql简单实现查询结果添加序列号的方法

    2024-01-26 11:33:56
  • 深入了解python列表(LIST)

    2022-09-04 18:25:55
  • Tensorflow 实现分批量读取数据

    2023-09-23 23:04:44
  • python中pygame安装过程(超级详细)

    2022-08-05 04:26:15
  • debug case: onsubmit=return false依然能提交

    2009-01-18 13:07:00
  • 浅谈Python如何获取excel数据

    2021-04-06 09:30:32
  • python实现图片批量剪切示例

    2021-11-18 13:04:38
  • python函数的重新定义及练习

    2023-10-12 22:47:55
  • Asp 编码互转的研究和实现代码

    2011-03-30 10:44:00
  • 纠结于ajax开发中 response的contentType 问题

    2008-12-11 13:46:00
  • caffe的python接口生成配置文件学习

    2023-07-09 04:46:41
  • asp之家 网络编程 m.aspxhome.com