欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品

主頁 > 知識庫 > Golang之sync.Pool使用詳解

Golang之sync.Pool使用詳解

熱門標簽:excel地圖標注分布數據 阿克蘇地圖標注 外呼系統顯本地手機號 百度地圖標注后傳給手機 外呼系統用什么卡 電話機器人軟件免費 涿州代理外呼系統 評價高的400電話辦理 壽光微信地圖標注

前言

我們通常用 Golang 來開發并構建高并發場景下的服務,但是由于 Golang 內建的GC機制多少會影響服務的性能,因此,為了減少頻繁GC,Golang提供了對象重用的機制,也就是使用sync.Pool構建對象池。

sync.Pool介紹

首先sync.Pool是可伸縮的臨時對象池,也是并發安全的。其可伸縮的大小會受限于內存的大小,可以理解為是一個存放可重用對象的容器。sync.Pool設計的目的就是用于存放已經分配的但是暫時又不用的對象,而且在需要用到的時候,可以直接從該pool中取。

pool中任何存放的值可以在任何時候被刪除而不會收到通知。另外,在高負載下pool對象池可以動態的擴容,而在不使用或者說并發量不高時對象池會收縮。關鍵思想就是對象的復用,避免重復創建、銷毀,從而影響性能。

個人覺得它的名字有一定的誤導性,因為 Pool 里裝的對象可以被無通知地被回收,覺得 sync.Cache 的名字更合適sync.Pool的命名。

sync.Pool首先聲明了兩個結構體,如下:

// Local per-P Pool appendix.
type poolLocalInternal struct {
  private interface{} // Can be used only by the respective P.
  shared  poolChain   // Local P can pushHead/popHead; any P can popTail.
}

type poolLocal struct {
  poolLocalInternal

  // Prevents false sharing on widespread platforms with
  // 128 mod (cache line size) = 0 .
  pad [128 - unsafe.Sizeof(poolLocalInternal{})%128]byte
}

為了使得可以在多個goroutine中高效的使用并發,sync.Pool會為每個P(對應CPU,這里有點像GMP模型)都分配一個本地池,當執行Get或者Put操作的時候,會先將goroutine和某個P的對象池關聯,再對該池進行操作。

每個P的對象池分為私有對象和共享列表對象,私有對象只能被特定的P訪問,共享列表對象可以被任何P訪問。因為同一時刻一個P只能執行一個goroutine,所以無需加鎖,但是對共享列表對象進行操作時,因為可能有多個goroutine同時操作,即并發操作,所以需要加鎖。

需要注意的是 poolLocal 結構體中有個 pad 成員,其目的是為了防止false sharing。cache使用中常見的一個問題是false sharing。當不同的線程同時讀寫同一個 cache line上不同數據時就可能發生false sharing。false sharing會導致多核處理器上嚴重的系統性能下降。具體的解釋說明這里就不展開贅述了。

sync.Pool的Put和Get方法

sync.Pool 有兩個公開的方法,一個是Get,另一個是Put。

Put方法

我們先來看一下Put方法的源碼,如下:

// Put adds x to the pool.
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
  }
  if x != nil {
    l.shared.pushHead(x)
  }
  runtime_procUnpin()
  if race.Enabled {
    race.Enable()
  }
}

閱讀以上Put方法的源碼可以知道:

  • 如果Put放入的值為空,則直接 return 了,不會執行下面的邏輯了;
  • 如果不為空,則繼續檢查當前goroutine的private是否設置對象池私有值,如果沒有則將x賦值給該私有成員,并將x設置為nil;
  • 如果當前goroutine的private私有值已經被賦值過了,那么將該值追加到共享列表。

Get方法

我們再來看下Get方法的源碼,如下:

func (p *Pool) Get() interface{} {
  if race.Enabled {
    race.Disable()
  }
  l, pid := p.pin()
  x := l.private
  l.private = nil
  if x == nil {
    // Try to pop the head of the local shard. We prefer
    // the head over the tail for temporal locality of
    // reuse.
    x, _ = l.shared.popHead()
    if x == nil {
      x = p.getSlow(pid)
    }
  }
  runtime_procUnpin()
  if race.Enabled {
    race.Enable()
    if x != nil {
      race.Acquire(poolRaceAddr(x))
    }
  }
  if x == nil  p.New != nil {
    x = p.New()
  }
  return x
}

閱讀以上Get方法的源碼,可以知道:

  • 首先嘗試從本地P對應的那個對象池中獲取一個對象值, 并從對象池中刪掉該值。
  • 如果從本地對象池中獲取失敗,則從共享列表中獲取,并從共享列表中刪除該值。
  • 如果從共享列表中獲取失敗,則會從其它P的對象池中“偷”一個過來,并刪除共享池中的該值(就是源碼中14行的p.getSlow())。
  • 如果還是失敗,那么直接通過 New() 分配一個返回值,注意這個分配的值不會被放入對象池中。New()是返回用戶注冊的New函數的值,如果用戶未注冊New,那么默認返回nil。

init函數

最后我們來看一下init函數,如下:

func init() {
  funtime_registerPoolCleanup(poolCleanup)
}

可以看到在init的時候注冊了一個PoolCleanup函數,他會清除掉sync.Pool中的所有的緩存的對象,這個注冊函數會在每次GC的時候運行,所以sync.Pool中的值只在兩次GC中間的時段有效。

sync.Pool使用示例

示例代碼:

package main
import (
 "fmt"
 "sync"
)
// 定義一個 Person 結構體,有Name和Age變量
type Person struct {
 Name string
 Age int
}
// 初始化sync.Pool,new函數就是創建Person結構體
func initPool() *sync.Pool {
 return sync.Pool{
  New: func() interface{} {
   fmt.Println("創建一個 person.")
   return Person{}
  },
 }
}
// 主函數,入口函數
func main() {
 pool := initPool()
 person := pool.Get().(*Person)
 fmt.Println("首次從sync.Pool中獲取person:", person)
 person.Name = "Jack"
 person.Age = 23
 pool.Put(person)
 fmt.Println("設置的對象Name: ", person.Name)
 fmt.Println("設置的對象Age: ", person.Age)
 fmt.Println("Pool 中有一個對象,調用Get方法獲取:", pool.Get().(*Person))
 fmt.Println("Pool 中沒有對象了,再次調用Get方法:", pool.Get().(*Person))
}

運行結果如下所示:

創建一個 person.
首次從sync.Pool中獲取person:{ 0}
設置的對象Name:  Jack
設置的對象Age:  23
Pool 中有一個對象,調用Get方法獲取:{Jack 23}
創建一個 person.
Pool 中沒有對象了,再次調用Get方法: { 0}

總結

通過以上的源碼及其示例,我們可以知道:

  • Get方法并不會對獲取到的對象值做任何的保證,因為放入本地對象池中的值有可能會在任何時候被刪除,而得不到通知。
  • 放入共享池中的值有可能被其他的goroutine拿走,所以對象池比較適合用來存儲一些臨時切狀態無關的數據,但是不適合用來存儲數據庫連接的實例,因為存入對象池的值有可能會在垃圾回收時被刪除掉,這違反了數據庫連接池建立的初衷。

由此可知,Golang的對象池嚴格意義上來說是一個臨時的對象池,適用于儲存一些會在goroutine間分享的臨時對象。主要作用是減少GC,提高性能。在Golang中最常見的使用場景就是fmt包中的輸出緩沖區了。

代碼Github歸檔地址: sync.Pool使用示例代碼

到此這篇關于Golang之sync.Pool使用詳解的文章就介紹到這了,更多相關Golang sync.Pool內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 深入Golang中的sync.Pool詳解

標簽:欽州 汕頭 吐魯番 銅川 蘭州 重慶 梅河口 雞西

巨人網絡通訊聲明:本文標題《Golang之sync.Pool使用詳解》,本文關鍵詞  Golang,之,sync.Pool,使用,詳解,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Golang之sync.Pool使用詳解》相關的同類信息!
  • 本頁收集關于Golang之sync.Pool使用詳解的相關信息資訊供網民參考!
  • 推薦文章
    欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品
  • <rt id="w000q"><acronym id="w000q"></acronym></rt>
  • <abbr id="w000q"></abbr>
    <rt id="w000q"></rt>
    成年人视频软件| 精品国产3级a| 亚洲欧洲另类国产综合| 国产一区二区三区国产| 亚洲精品国产91| 精品久久久久久综合日本欧美| 日韩va亚洲va欧美va久久| 精品影片一区二区入口| 欧美顶级少妇做爰| 午夜精品在线视频一区| 日本五十肥熟交尾| 91精品啪在线观看国产60岁| 视频在线在亚洲| 久久人人爽人人爽人人片| 日韩欧美一区二区不卡| 全部av―极品视觉盛宴亚洲| 无码h肉动漫在线观看| 精品久久久久久久久久久久久久久| 蜜臀精品久久久久久蜜臀| 女女互磨互喷水高潮les呻吟| 精品sm在线观看| 国产老妇另类xxxxx| 色欲一区二区三区精品a片| 综合久久久久综合| 国产精品熟女一区二区不卡| 欧美美女一区二区在线观看| 首页国产丝袜综合| 亚洲av无码国产精品麻豆天美| 精品久久人人做人人爰| 国产精品一品二品| 国产这里有精品| 亚洲一区二区偷拍精品| 97人妻天天摸天天爽天天| 久久影院午夜片一区| 国产成人免费xxxxxxxx| 欧洲国产伦久久久久久久| 亚洲成av人片观看| 韩国三级hd中文字幕| 国产精品区一区二区三| 91毛片在线观看| 日韩精品综合一本久道在线视频| 国产在线一区二区| 日本韩国一区二区| 视频在线观看一区| 国产日产在线观看| 玉足女爽爽91| 亚洲欧美韩国综合色| 艳妇乳肉豪妇荡乳xxx| 久久久亚洲精品石原莉奈| 成人禁用看黄a在线| 欧美男男青年gay1069videost| 精品一区二区三区在线观看| 色综合色狠狠天天综合色| 天天综合色天天综合色h| 国产精品一区二区亚洲| 亚洲最快最全在线视频| 自拍偷拍视频亚洲| 亚洲人成精品久久久久| 91精品国产自产| 一区二区中文视频| 熟妇高潮精品一区二区三区| 国产精品网站在线| 欧美大喷水吹潮合集在线观看| 国产午夜精品美女毛片视频| 91蝌蚪porny九色| 欧美精品一区二区三区蜜桃| 91麻豆精品一区二区三区| 欧美精品一区二区三区久久久 | 亚洲女优在线观看| 亚洲啪啪综合av一区二区三区| 国产黄色网址在线观看| 亚洲欧洲无码一区二区三区| 免费黄色在线视频| 一区二区三区鲁丝不卡| 九九九视频在线观看| 亚洲成av人**亚洲成av**| 亚洲伦理一区二区三区| 日本aⅴ精品一区二区三区| 色哟哟国产精品| 国产综合久久久久久久久久久久 | 国产精品69毛片高清亚洲| 欧美日韩aaaaa| 成人免费av网站| 337p日本欧洲亚洲大胆精品 | 国产麻豆91精品| 加勒比精品视频| 国产精品二三区| xxxxx在线观看| 亚洲国产精品久久久久婷婷884| 欧美成人久久久免费播放| 偷偷要91色婷婷| 欧美一a一片一级一片| 国产精一品亚洲二区在线视频| 欧美一区二区三区视频在线| 91色综合久久久久婷婷| 中文成人综合网| 超薄肉色丝袜一二三| 日本伊人色综合网| 欧美日韩精品是欧美日韩精品| 不卡av在线免费观看| 国产蜜臀97一区二区三区| 成人午夜福利一区二区| 天堂成人免费av电影一区| 欧美三级在线视频| 99久久精品免费看国产免费软件| 国产精品每日更新在线播放网址 | 亚洲国产精品欧美一二99| 26uuu成人网| 国产精品一区二区三区网站| 欧美精品一区二| 丰满少妇一区二区| 男女性色大片免费观看一区二区| 欧美日韩中文字幕一区二区| bt欧美亚洲午夜电影天堂| 国产精品久久毛片av大全日韩| 一级片久久久久| 狠狠色丁香久久婷婷综合_中| 欧美刺激脚交jootjob| 国产精品无码在线| 日韩精品一区第一页| 91精品午夜视频| 天天插天天射天天干| 日韩激情在线观看| 午夜av一区二区三区| 欧美日韩一区不卡| 中文字幕一二三区| 一区二区欧美国产| 欧洲精品在线观看| 人妻av一区二区三区| 亚洲成人动漫在线免费观看| 欧美喷潮久久久xxxxx| 男女性杂交内射妇女bbwxz| 亚洲成人精品一区| 欧美日韩一级大片网址| 国产精品久久久久久亚洲色| 日韩av一区二区三区四区| 欧美成人激情免费网| 女女互磨互喷水高潮les呻吟| 韩国一区二区三区| 国产拍欧美日韩视频二区| 青花影视在线观看免费高清| youjizz久久| 亚洲午夜在线电影| 欧美一区二区久久久| 亚洲午夜久久久久久久国产| 国产精品一区二区三区乱码| 综合亚洲深深色噜噜狠狠网站| 欧美色视频在线| jizz日本免费| 国产呦精品一区二区三区网站| 中文av一区二区| 欧美性三三影院| 精品人妻一区二区三区日产乱码卜| 另类中文字幕网| 国产精品久久久久三级| 欧美性videosxxxxx| 三级电影在线看| 国产精品乡下勾搭老头1| 亚洲欧美一区二区三区孕妇| 欧美精品123区| 日本少妇xxxxx| 99久久久久免费精品国产| 午夜亚洲国产au精品一区二区 | 精产国品一区二区三区| 日韩高清一级片| 天天看片中文字幕| 久久精品免费观看| 国产精品五月天| 欧美老年两性高潮| xxxx日本黄色| 9久草视频在线视频精品| 天天av天天翘天天综合网| 国产亚洲欧美日韩在线一区| 91久久线看在观草草青青| 国产人妻人伦精品1国产丝袜| 国产精品99久久久久久久女警 | av网站免费线看精品| 五月天激情小说综合| 国产婷婷色一区二区三区四区| 91久久精品午夜一区二区| 国内精品久久99人妻无码| 成人国产精品免费观看动漫| 天天色天天操综合| 国产精品成人一区二区三区夜夜夜| 欧美高清视频一二三区| 日本一级特级毛片视频| 成人性生活免费看| 成人中文字幕在线| 亚洲国产日韩在线一区模特| 国产亚洲欧美一区在线观看| 欧美精品丝袜中出| 午夜爱爱毛片xxxx视频免费看| 亚洲调教欧美在线| 97se狠狠狠综合亚洲狠狠| 久久av中文字幕片| 亚洲一二三级电影| 国产精品久久久久久久第一福利| 欧美一区二区视频在线观看 | 日韩一级片av| 色无极影院亚洲|