前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >修改golang源代码获取goroutine id实现ThreadLocal

修改golang源代码获取goroutine id实现ThreadLocal

作者头像
左手java右手go
发布2019-08-20 15:54:14
1.1K0
发布2019-08-20 15:54:14
举报

开篇

golang在http.Request中提供了一个Context用于存储kv对,我们可以通过这个来存储请求相关的数据。在请求入口,我们把唯一的requstID存储到context中,在后续需要调用的地方把值取出来打印。如果日志是在controller中打印,这个很好处理,http.Request是作为入参的。但如果是在更底层呢?比如说是在model甚至是一些工具类中。我们当然可以给每个方法都提供一个参数,由调用方把context一层一层传下来,但这种方式明显不够优雅。想想java里面是怎么做的--ThreadLocal。虽然golang官方不太认可这种方式,但是我们今天就是要基于goroutine id实现它。

We wouldn't even be having this discussion if thread local storage wasn't useful. But every feature comes at a cost, and in my opinion the cost of threadlocals far outweighs their benefits. They're just not a good fit for Go.

思路

每个goroutine有一个唯一的id,但是被隐藏了,我们首先把它暴露出来,然后建立一个map,用id作为key,goroutineLocal存储的实际数据作为value。

获取goroutine id

1.修改 $GOROOT/src/runtime/proc.go 文件,添加 GetGoroutineId() 函数

func GetGoroutineId() int64 {    return getg().goid
}

其中getg()函数是获取当前执行的g对象,g对象包含了栈,cgo信息,GC信息,goid等相关数据,goid就是我们想要的。

2.重新编译源码

cd ~/go/src
GOROOT_BOOTSTRAP='/Users/qiuxudong/go1.9' ./all.bash

实现 GoroutineLocal

package goroutine_localimport (    "sync"
   "runtime")type goroutineLocal struct {    initfun func() interface{}
   m *sync.Map
}func NewGoroutineLocal(initfun func() interface{}) *goroutineLocal {    return &goroutineLocal{initfun:initfun, m:&sync.Map{}}
}func (gl *goroutineLocal)Get() interface{} {
   value, ok := gl.m.Load(runtime.GetGoroutineId())    if !ok && gl.initfun != nil {
       value = gl.initfun()
   }    return value
}func (gl *goroutineLocal)Set(v interface{}) {
   gl.m.Store(runtime.GetGoroutineId(), v)
}func (gl *goroutineLocal)Remove() {
   gl.m.Delete(runtime.GetGoroutineId())
}

简单测试一下

package goroutine_localimport (    "testing"
   "fmt"
   "time"
   "runtime")var gl = NewGoroutineLocal(func() interface{} {    return "default"})func TestGoroutineLocal(t *testing.T) {
   gl.Set("test0")
   fmt.Println(runtime.GetGoroutineId(), gl.Get())    go func() {
       gl.Set("test1")
       fmt.Println(runtime.GetGoroutineId(), gl.Get())
       gl.Remove()
       fmt.Println(runtime.GetGoroutineId(), gl.Get())
   }()   time.Sleep(2 * time.Second)
}

可以看到结果

5 test06 test16 default

内存泄露问题

由于跟goroutine绑定的数据放在goroutineLocal的map里面,即使goroutine销毁了数据还在,可能存在内存泄露,因此不使用时要记得调用Remove清除数据

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-02-12,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 左手java右手go 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 开篇
  • 思路
  • 获取goroutine id
  • 实现 GoroutineLocal
  • 内存泄露问题
相关产品与服务
对象存储
对象存储(Cloud Object Storage,COS)是由腾讯云推出的无目录层次结构、无数据格式限制,可容纳海量数据且支持 HTTP/HTTPS 协议访问的分布式存储服务。腾讯云 COS 的存储桶空间无容量上限,无需分区管理,适用于 CDN 数据分发、数据万象处理或大数据计算与分析的数据湖等多种场景。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档