這篇文章給大家分享的是有關Go臨時對象池pool怎么用的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
創新互聯公司自2013年起,是專業互聯網技術服務公司,擁有項目成都網站設計、網站建設網站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元撫松做網站,已為上家服務,為撫松各地企業和個人服務,聯系電話:18982081108臨時對象池 pool 是啥?
sync.Pool 給了一大段注釋來說明 pool 是啥,我們看看這段都說了些什么。
臨時對象池是一些可以分別存儲和取出的臨時對象。
池中的對象會在沒有任何通知的情況下被移出(釋放或者重新取出使用)。如果 pool 中持有某個對象的唯一引用,則該對象很可能會被回收。
Pool 在多 goroutine 使用環境中是安全的。
Pool 是用來緩存已經申請了的 目前未使用的 接下來可能會使用的 內存,以此緩解 GC 壓力。使用它可以方便高效的構建線程安全的 free list(一種用于動態內存申請的數據結構)。然而,它并不適合所有場景的 free list。
在同一 package 中獨立運行的多個獨立線程之間靜默共享一組臨時元素才是 pool 的合理使用場景。Pool 提供在多個獨立 client 之間共享臨時元素的機制。
在 fmt 包中有一個使用 Pool 的例子,它維護了一個動態大小的輸出 buffer。
另外,一些短生命周期的對象不適合使用 pool 來維護,這種情況下使用 pool 不劃算。這是應該使用它們自己的 free list(這里可能指的是 go 內存模型中用于緩存 <32k小對象的 free list) 更高效。
Pool 一旦使用,不能被復制。
Pool 結構體的定義為:
type Pool struct { noCopy noCopy local unsafe.Pointer // 本地P緩存池指針 localSize uintptr // 本地P緩存池大小 // 當池中沒有可能對象時 // 會調用 New 函數構造構造一個對象 New func() interface{} }
Pool 中有兩個定義的公共方法,分別是 Put - 向池中添加元素;Get - 從池中獲取元素,如果沒有,則調用 New 生成元素,如果 New 未設置,則返回 nil。
Get
Pool 會為每個 P 維護一個本地池,P 的本地池分為 私有池 private 和共享池 shared。私有池中的元素只能本地 P 使用,共享池中的元素可能會被其他 P 偷走,所以使用私有池 private 時不用加鎖,而使用共享池 shared 時需加鎖。
Get 會優先查找本地 private,再查找本地 shared,最后查找其他 P 的 shared,如果以上全部沒有可用元素,最后會調用 New 函數獲取新元素。
func (p *Pool) Get() interface{} { if race.Enabled { race.Disable() } // 獲取本地 P 的 poolLocal 對象 l := p.pin() // 先獲取 private 池中的對象(只有一個) x := l.private l.private = nil runtime_procUnpin() if x == nil { // 查找本地 shared 池, // 本地 shared 可能會被其他 P 訪問 // 需要加鎖 l.Lock() last := len(l.shared) - 1 if last >= 0 { x = l.shared[last] l.shared = l.shared[:last] } l.Unlock() // 查找其他 P 的 shared 池 if x == nil { x = p.getSlow() } } if race.Enabled { race.Enable() if x != nil { race.Acquire(poolRaceAddr(x)) } } // 未找到可用元素,調用 New 生成 if x == nil && p.New != nil { x = p.New() } return x }
getSlow,從其他 P 中的 shared 池中獲取可用元素:
func (p *Pool) getSlow() (x interface{}) { // See the comment in pin regarding ordering of the loads. size := atomic.LoadUintptr(&p.localSize) // load-acquire local := p.local // load-consume // Try to steal one element from other procs. pid := runtime_procPin() runtime_procUnpin() for i := 0; i < int(size); i++ { l := indexLocal(local, (pid+i+1)%int(size)) // 對應 pool 需加鎖 l.Lock() last := len(l.shared) - 1 if last >= 0 { x = l.shared[last] l.shared = l.shared[:last] l.Unlock() break } l.Unlock() } return x }
Put
Put 優先把元素放在 private 池中;如果 private 不為空,則放在 shared 池中。有趣的是,在入池之前,該元素有 1/4 可能被丟掉。
func (p *Pool) Put(x interface{}) { if x == nil { return } if race.Enabled { if fastrand()%4 == 0 { // 隨機把元素扔掉... // Randomly drop x on floor. return } race.ReleaseMerge(poolRaceAddr(x)) race.Disable() } l := p.pin() if l.private == nil { l.private = x x = nil } runtime_procUnpin() if x != nil { // 共享池訪問,需要加鎖 l.Lock() l.shared = append(l.shared, x) l.Unlock() } if race.Enabled { race.Enable() } }
poolCleanup
當世界暫停,垃圾回收將要開始時, poolCleanup 會被調用。該函數內不能分配內存且不能調用任何運行時函數。原因:
防止錯誤的保留整個 Pool
如果 GC 發生時,某個 goroutine 正在訪問 l.shared,整個 Pool 將會保留,下次執行時將會有雙倍內存
func poolCleanup() { for i, p := range allPools { allPools[i] = nil for i := 0; i < int(p.localSize); i++ { l := indexLocal(p.local, i) l.private = nil for j := range l.shared { l.shared[j] = nil } l.shared = nil } p.local = nil p.localSize = 0 } allPools = []*Pool{} }
案例1:gin 中的 Context pool
在 web 應用中,后臺在處理用戶的每條請求時都會為當前請求創建一個上下文環境 Context,用于存儲請求信息及相應信息等。Context 滿足長生命周期的特點,且用戶請求也是屬于并發環境,所以對于線程安全的 Pool 非常適合用來維護 Context 的臨時對象池。
Gin 在結構體 Engine 中定義了一個 pool:
type Engine struct { // ... 省略了其他字段 pool sync.Pool }
初始化 engine 時定義了 pool 的 New 函數:
engine.pool.New = func() interface{} { return engine.allocateContext() } // allocateContext func (engine *Engine) allocateContext() *Context { // 構造新的上下文對象 return &Context{engine: engine} }
ServeHttp:
// 從 pool 中獲取,并轉化為 *Context c := engine.pool.Get().(*Context) c.writermem.reset(w) c.Request = req c.reset() // reset engine.handleHTTPRequest(c) // 再扔回 pool 中 engine.pool.Put(c)
案例2:fmt 中的 printer pool
printer 也符合長生命周期的特點,同時也會可能會在多 goroutine 中使用,所以也適合使用 pool 來維護。
printer 與 它的臨時對象池
// pp 用來維護 printer 的狀態 // 它通過 sync.Pool 來重用,避免申請內存 type pp struct { //... 字段已省略 } var ppFree = sync.Pool{ New: func() interface{} { return new(pp) }, }
獲取與釋放:
func newPrinter() *pp { p := ppFree.Get().(*pp) p.panicking = false p.erroring = false p.fmt.init(&p.buf) return p } func (p *pp) free() { p.buf = p.buf[:0] p.arg = nil p.value = reflect.Value{} ppFree.Put(p) }
感謝各位的閱讀!關于“Go臨時對象池pool怎么用”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!
另外有需要云服務器可以了解下創新互聯scvps.cn,海內外云服務器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務器、裸金屬服務器、高防服務器、香港服務器、美國服務器、虛擬主機、免備案服務器”等云主機租用服務以及企業上云的綜合解決方案,具有“安全穩定、簡單易用、服務可用性高、性價比高”等特點與優勢,專為企業上云打造定制,能夠滿足用戶豐富、多元化的應用場景需求。
當前標題:Go臨時對象池pool怎么用-創新互聯
分享地址:http://vcdvsql.cn/article24/hspce.html
成都網站建設公司_創新互聯,為您提供品牌網站設計、網站維護、動態網站、域名注冊、品牌網站制作、網站建設
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯