修改golang源代碼獲取goroutine id實(shí)現(xiàn)ThreadLocal

開篇

golang在http.Request中提供了一個(gè)Context用于存儲(chǔ)kv對,我們可以通過這個(gè)來存儲(chǔ)請求相關(guān)的數(shù)據(jù)。在請求入口摊册,我們把唯一的requstID存儲(chǔ)到context中,在后續(xù)需要調(diào)用的地方把值取出來打印颊艳。如果日志是在controller中打印丧靡,這個(gè)很好處理,http.Request是作為入?yún)⒌淖严尽5绻窃诟讓幽匚轮危勘热缯f是在model甚至是一些工具類中。我們當(dāng)然可以給每個(gè)方法都提供一個(gè)參數(shù)戒悠,由調(diào)用方把context一層一層傳下來熬荆,但這種方式明顯不夠優(yōu)雅。想想java里面是怎么做的--ThreadLocal绸狐。雖然golang官方不太認(rèn)可這種方式卤恳,但是我們今天就是要基于goroutine id實(shí)現(xiàn)它累盗。

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.

思路

每個(gè)goroutine有一個(gè)唯一的id,但是被隱藏了突琳,我們首先把它暴露出來若债,然后建立一個(gè)map,用id作為key拆融,goroutineLocal存儲(chǔ)的實(shí)際數(shù)據(jù)作為value蠢琳。

獲取goroutine id

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

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

其中g(shù)etg()函數(shù)是獲取當(dāng)前執(zhí)行的g對象镜豹,g對象包含了棧傲须,cgo信息,GC信息趟脂,goid等相關(guān)數(shù)據(jù)泰讽,goid就是我們想要的。

2.重新編譯源碼

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

實(shí)現(xiàn) GoroutineLocal

package goroutine_local

import (
    "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_local

import (
    "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)
}

可以看到結(jié)果

5 test0
6 test1
6 default

內(nèi)存泄露問題

由于跟goroutine綁定的數(shù)據(jù)放在goroutineLocal的map里面昔期,即使goroutine銷毀了數(shù)據(jù)還在已卸,可能存在內(nèi)存泄露,因此不使用時(shí)要記得調(diào)用Remove清除數(shù)據(jù)

歡迎您掃一掃上面的二維碼關(guān)注個(gè)人微信公眾號(hào)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末硼一,一起剝皮案震驚了整個(gè)濱河市累澡,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌欠动,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,378評(píng)論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件惑申,死亡現(xiàn)場離奇詭異具伍,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)圈驼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,970評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門人芽,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人绩脆,你說我怎么就攤上這事萤厅。” “怎么了靴迫?”我有些...
    開封第一講書人閱讀 168,983評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵惕味,是天一觀的道長。 經(jīng)常有香客問我玉锌,道長名挥,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,938評(píng)論 1 299
  • 正文 為了忘掉前任主守,我火速辦了婚禮禀倔,結(jié)果婚禮上榄融,老公的妹妹穿的比我還像新娘。我一直安慰自己救湖,他們只是感情好愧杯,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,955評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著鞋既,像睡著了一般力九。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上涛救,一...
    開封第一講書人閱讀 52,549評(píng)論 1 312
  • 那天畏邢,我揣著相機(jī)與錄音,去河邊找鬼检吆。 笑死舒萎,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的蹭沛。 我是一名探鬼主播臂寝,決...
    沈念sama閱讀 41,063評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼摊灭!你這毒婦竟也來了咆贬?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,991評(píng)論 0 277
  • 序言:老撾萬榮一對情侶失蹤帚呼,失蹤者是張志新(化名)和其女友劉穎掏缎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體煤杀,經(jīng)...
    沈念sama閱讀 46,522評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡眷蜈,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,604評(píng)論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了沈自。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片酌儒。...
    茶點(diǎn)故事閱讀 40,742評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖枯途,靈堂內(nèi)的尸體忽然破棺而出忌怎,到底是詐尸還是另有隱情,我是刑警寧澤酪夷,帶...
    沈念sama閱讀 36,413評(píng)論 5 351
  • 正文 年R本政府宣布榴啸,位于F島的核電站,受9級(jí)特大地震影響晚岭,放射性物質(zhì)發(fā)生泄漏插掂。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,094評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望辅甥。 院中可真熱鬧酝润,春花似錦、人聲如沸璃弄。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,572評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽夏块。三九已至疏咐,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間脐供,已是汗流浹背浑塞。 一陣腳步聲響...
    開封第一講書人閱讀 33,671評(píng)論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留政己,地道東北人酌壕。 一個(gè)月前我還...
    沈念sama閱讀 49,159評(píng)論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像歇由,于是被迫代替她去往敵國和親卵牍。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,747評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容