Go的context是线程安全的吗

本文探讨了Go语言中的context是如何实现线程安全的。通过分析`context.WithValue`函数,可以看出每次调用都会创建一个新的valueCtx子节点,形成context链,确保键值对的添加不会修改原有context,因此是线程安全的。context包提供了Background和TODO方法创建根context,并通过Value、Deadline、Done、Err方法衍生新的context。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

context本身是线程安全的,所以context携带value也是线程安全的.

func main()  {
 ctx := context.WithValue(context.Background(), "key", "value01")
 go func() {
  for {
   _ = context.WithValue(ctx, "key", "value02")
  }
 }()
 go func() {
  for {
   _ = context.WithValue(ctx, "key", "value03")
  }
 }()
 go func() {
  for {
   fmt.Println(ctx.Value("key"))
  }
 }()
 go func() {
  for {
   fmt.Println(ctx.Value("key"))
  }
 }()
 time.Sleep(3 * time.Second)
}

程序是没有问题的,接下来就来看一下为什么context是线程安全的.

为什么线程安全?

context包提供两种创建根context的方式:

  • context.Backgroud()

  • context.TODO()

又提供了四个函数(Value、Deadline、Done、Err)基于父Context衍生,其中使用WithValue函数来衍生context并携带数据,每次调用WithValue函数都会基于当前context衍生一个新的子contextWithValue内部主要就是调用valueCtx类:

func WithValue(parent Context, key, val interface{}) Context {
	if key == nil {
		panic("nil key")
	}
	if !reflectlite.TypeOf(key).Comparable() {
		panic("key is not comparable")
	}
	return &valueCtx{parent, key, val}
}

说明:这里的parent是当前valueContext的父节点。

valueCtx结构如下:

type valueCtx struct {
 Context
 key, val interface{}
}

valueContext继承父Context,这种是采用匿名接口的继承实现方式,key、val用来存储携带的键值对。

通过上面的代码分析,可以看到添加键值对不是在原Context结构体上直接添加,而是以此context作为父节点,重新创建一个新的valueContext子节点,将键值对添加到子节点上,由此形成一条context链。

获取键值对的过程也是层层向上调用直到首次设置key的父节点,如果没有找到首次设置key的父节点,会向上遍历直到根节点,如果根节点找到了key就会返回,否则就会找到最终的emptyCtx返回nil。如下图所示:

总结:context添加的键值对是一个链式的,会不断衍生新的context,所以context本身是不可变的,因此是线程安全的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值