亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識(shí)庫 > Golang之sync.Pool使用詳解

Golang之sync.Pool使用詳解

熱門標(biāo)簽:excel地圖標(biāo)注分布數(shù)據(jù) 阿克蘇地圖標(biāo)注 外呼系統(tǒng)顯本地手機(jī)號(hào) 百度地圖標(biāo)注后傳給手機(jī) 外呼系統(tǒng)用什么卡 電話機(jī)器人軟件免費(fèi) 涿州代理外呼系統(tǒng) 評(píng)價(jià)高的400電話辦理 壽光微信地圖標(biāo)注

前言

我們通常用 Golang 來開發(fā)并構(gòu)建高并發(fā)場(chǎng)景下的服務(wù),但是由于 Golang 內(nèi)建的GC機(jī)制多少會(huì)影響服務(wù)的性能,因此,為了減少頻繁GC,Golang提供了對(duì)象重用的機(jī)制,也就是使用sync.Pool構(gòu)建對(duì)象池。

sync.Pool介紹

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

pool中任何存放的值可以在任何時(shí)候被刪除而不會(huì)收到通知。另外,在高負(fù)載下pool對(duì)象池可以動(dòng)態(tài)的擴(kuò)容,而在不使用或者說并發(fā)量不高時(shí)對(duì)象池會(huì)收縮。關(guān)鍵思想就是對(duì)象的復(fù)用,避免重復(fù)創(chuàng)建、銷毀,從而影響性能。

個(gè)人覺得它的名字有一定的誤導(dǎo)性,因?yàn)?Pool 里裝的對(duì)象可以被無通知地被回收,覺得 sync.Cache 的名字更合適sync.Pool的命名。

sync.Pool首先聲明了兩個(gè)結(jié)構(gòu)體,如下:

// 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
}

為了使得可以在多個(gè)goroutine中高效的使用并發(fā),sync.Pool會(huì)為每個(gè)P(對(duì)應(yīng)CPU,這里有點(diǎn)像GMP模型)都分配一個(gè)本地池,當(dāng)執(zhí)行Get或者Put操作的時(shí)候,會(huì)先將goroutine和某個(gè)P的對(duì)象池關(guān)聯(lián),再對(duì)該池進(jìn)行操作。

每個(gè)P的對(duì)象池分為私有對(duì)象和共享列表對(duì)象,私有對(duì)象只能被特定的P訪問,共享列表對(duì)象可以被任何P訪問。因?yàn)橥粫r(shí)刻一個(gè)P只能執(zhí)行一個(gè)goroutine,所以無需加鎖,但是對(duì)共享列表對(duì)象進(jìn)行操作時(shí),因?yàn)榭赡苡卸鄠€(gè)goroutine同時(shí)操作,即并發(fā)操作,所以需要加鎖。

需要注意的是 poolLocal 結(jié)構(gòu)體中有個(gè) pad 成員,其目的是為了防止false sharing。cache使用中常見的一個(gè)問題是false sharing。當(dāng)不同的線程同時(shí)讀寫同一個(gè) cache line上不同數(shù)據(jù)時(shí)就可能發(fā)生false sharing。false sharing會(huì)導(dǎo)致多核處理器上嚴(yán)重的系統(tǒng)性能下降。具體的解釋說明這里就不展開贅述了。

sync.Pool的Put和Get方法

sync.Pool 有兩個(gè)公開的方法,一個(gè)是Get,另一個(gè)是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 了,不會(huì)執(zhí)行下面的邏輯了;
  • 如果不為空,則繼續(xù)檢查當(dāng)前goroutine的private是否設(shè)置對(duì)象池私有值,如果沒有則將x賦值給該私有成員,并將x設(shè)置為nil;
  • 如果當(dāng)前goroutine的private私有值已經(jīng)被賦值過了,那么將該值追加到共享列表。

Get方法

我們?cè)賮砜聪翯et方法的源碼,如下:

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對(duì)應(yīng)的那個(gè)對(duì)象池中獲取一個(gè)對(duì)象值, 并從對(duì)象池中刪掉該值。
  • 如果從本地對(duì)象池中獲取失敗,則從共享列表中獲取,并從共享列表中刪除該值。
  • 如果從共享列表中獲取失敗,則會(huì)從其它P的對(duì)象池中“偷”一個(gè)過來,并刪除共享池中的該值(就是源碼中14行的p.getSlow())。
  • 如果還是失敗,那么直接通過 New() 分配一個(gè)返回值,注意這個(gè)分配的值不會(huì)被放入對(duì)象池中。New()是返回用戶注冊(cè)的New函數(shù)的值,如果用戶未注冊(cè)New,那么默認(rèn)返回nil。

init函數(shù)

最后我們來看一下init函數(shù),如下:

func init() {
  funtime_registerPoolCleanup(poolCleanup)
}

可以看到在init的時(shí)候注冊(cè)了一個(gè)PoolCleanup函數(shù),他會(huì)清除掉sync.Pool中的所有的緩存的對(duì)象,這個(gè)注冊(cè)函數(shù)會(huì)在每次GC的時(shí)候運(yùn)行,所以sync.Pool中的值只在兩次GC中間的時(shí)段有效。

sync.Pool使用示例

示例代碼:

package main
import (
 "fmt"
 "sync"
)
// 定義一個(gè) Person 結(jié)構(gòu)體,有Name和Age變量
type Person struct {
 Name string
 Age int
}
// 初始化sync.Pool,new函數(shù)就是創(chuàng)建Person結(jié)構(gòu)體
func initPool() *sync.Pool {
 return sync.Pool{
  New: func() interface{} {
   fmt.Println("創(chuàng)建一個(gè) person.")
   return Person{}
  },
 }
}
// 主函數(shù),入口函數(shù)
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("設(shè)置的對(duì)象Name: ", person.Name)
 fmt.Println("設(shè)置的對(duì)象Age: ", person.Age)
 fmt.Println("Pool 中有一個(gè)對(duì)象,調(diào)用Get方法獲取:", pool.Get().(*Person))
 fmt.Println("Pool 中沒有對(duì)象了,再次調(diào)用Get方法:", pool.Get().(*Person))
}

運(yùn)行結(jié)果如下所示:

創(chuàng)建一個(gè) person.
首次從sync.Pool中獲取person:{ 0}
設(shè)置的對(duì)象Name:  Jack
設(shè)置的對(duì)象Age:  23
Pool 中有一個(gè)對(duì)象,調(diào)用Get方法獲?。簕Jack 23}
創(chuàng)建一個(gè) person.
Pool 中沒有對(duì)象了,再次調(diào)用Get方法: { 0}

總結(jié)

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

  • Get方法并不會(huì)對(duì)獲取到的對(duì)象值做任何的保證,因?yàn)榉湃氡镜貙?duì)象池中的值有可能會(huì)在任何時(shí)候被刪除,而得不到通知。
  • 放入共享池中的值有可能被其他的goroutine拿走,所以對(duì)象池比較適合用來存儲(chǔ)一些臨時(shí)切狀態(tài)無關(guān)的數(shù)據(jù),但是不適合用來存儲(chǔ)數(shù)據(jù)庫連接的實(shí)例,因?yàn)榇嫒雽?duì)象池的值有可能會(huì)在垃圾回收時(shí)被刪除掉,這違反了數(shù)據(jù)庫連接池建立的初衷。

由此可知,Golang的對(duì)象池嚴(yán)格意義上來說是一個(gè)臨時(shí)的對(duì)象池,適用于儲(chǔ)存一些會(huì)在goroutine間分享的臨時(shí)對(duì)象。主要作用是減少GC,提高性能。在Golang中最常見的使用場(chǎng)景就是fmt包中的輸出緩沖區(qū)了。

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

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

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

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

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Golang之sync.Pool使用詳解》,本文關(guān)鍵詞  Golang,之,sync.Pool,使用,詳解,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Golang之sync.Pool使用詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于Golang之sync.Pool使用詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    日韩avdvd| 91麻豆精品国产片在线观看| 日本在线www| 成人影院一区二区三区| 成人高清护士在线播放| 日本特黄特色aa大片免费| 国产不卡精品一区二区三区| 91麻豆精品国产片在线观看| 欧美激情影院| 午夜久久网| 国产伦久视频免费观看视频| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 九九干| 日韩一级黄色大片| 国产成人精品在线| 黄视频网站在线看| 精品国产一区二区三区久久久狼 | 国产高清在线精品一区二区 | a级毛片免费全部播放| 国产不卡在线观看视频| 精品视频免费观看| 欧美电影免费看大全| 午夜在线亚洲| 精品久久久久久中文字幕一区| 国产一区免费在线观看| 一级女性全黄生活片免费| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 九九国产| 成人免费一级纶理片| 四虎久久影院| 91麻豆tv| 尤物视频网站在线观看| 青草国产在线| 久久福利影视| 一本高清在线| 国产综合成人观看在线| 亚欧成人乱码一区二区| 免费国产在线观看不卡| 成人免费一级纶理片| 九九精品久久| 毛片的网站| 九九免费高清在线观看视频| 精品视频一区二区| 国产视频一区二区在线观看| 国产麻豆精品hdvideoss| 国产91精品系列在线观看| 国产91视频网| 国产国语对白一级毛片| 国产91视频网| 久草免费资源| 精品国产一区二区三区精东影业| 精品国产一区二区三区久久久蜜臀 | 天天做日日爱| 黄视频网站在线看| 精品国产亚一区二区三区| 精品视频在线观看视频免费视频 | 精品视频免费观看| 国产综合91天堂亚洲国产| 天天做日日爱夜夜爽| 亚飞与亚基在线观看| 二级特黄绝大片免费视频大片| 国产亚洲免费观看| 日本乱中文字幕系列| 日本免费区| 999久久久免费精品国产牛牛| 国产网站麻豆精品视频| 欧美激情一区二区三区中文字幕| 天天色成人| 九九久久99综合一区二区| 精品国产一区二区三区久久久蜜臀 | 美女免费毛片| 日本特黄特色aaa大片免费| 精品视频在线观看免费| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 精品久久久久久中文| 韩国毛片| 99色视频在线| 日韩男人天堂| 99热精品在线| 亚洲女初尝黑人巨高清在线观看| 尤物视频网站在线观看| 精品视频在线观看一区二区| 久久精品免视看国产明星| 国产精品免费久久| 一级片免费在线观看视频| 亚州视频一区二区| 亚洲 男人 天堂| 99色吧| 国产一区二区福利久久| 国产不卡高清在线观看视频| 超级乱淫黄漫画免费| 九九热精品免费观看| 一级女性全黄生活片免费| 成人免费福利片在线观看| 亚洲女人国产香蕉久久精品 | 九九精品久久久久久久久| 你懂的日韩| 亚洲天堂在线播放| 人人干人人草| 台湾美女古装一级毛片| 国产视频一区二区在线观看| 国产视频一区在线| 99久久精品国产麻豆| 色综合久久天天综合绕观看| 日韩免费片| 99久久精品国产片| 国产一区二区精品在线观看| 一级女性大黄生活片免费| 麻豆午夜视频| 日韩在线观看视频黄| 国产伦久视频免费观看视频| a级精品九九九大片免费看| 天天做人人爱夜夜爽2020| 成人影院久久久久久影院| 成人a大片在线观看| 99色视频在线观看| 九九九网站| 国产亚洲精品成人a在线| 精品国产一区二区三区久久久狼 | 黄视频网站在线看| 99热精品在线| 中文字幕一区二区三区 精品| 青青久久精品国产免费看| 欧美激情一区二区三区视频| 国产a网| 欧美激情一区二区三区在线| 精品视频在线观看一区二区| 日日爽天天| 国产一区二区福利久久| 黄色免费三级| 欧美一级视频免费| 国产网站免费视频| 毛片高清| 欧美另类videosbestsex| 午夜在线亚洲男人午在线| 可以免费看污视频的网站| 美女免费精品高清毛片在线视| 美女免费精品高清毛片在线视| 91麻豆精品国产片在线观看| 你懂的福利视频| 九九久久99综合一区二区| 免费国产在线观看不卡| 日韩在线观看视频免费| 亚洲 国产精品 日韩| 亚洲第一色在线| 午夜在线亚洲| 国产麻豆精品hdvideoss| 中文字幕一区二区三区 精品| 国产麻豆精品高清在线播放| 99色视频在线| 欧美激情中文字幕一区二区| 国产麻豆精品| 黄视频网站在线免费观看| 在线观看导航| 久久99中文字幕久久| 色综合久久天天综合绕观看| 久久精品免视看国产明星| 黄色免费网站在线| 91麻豆精品国产片在线观看| 久久精品免视看国产明星| 精品国产一区二区三区国产馆| 人人干人人草| 免费一级片在线| 欧美国产日韩一区二区三区| 亚洲精品影院久久久久久| 999久久66久6只有精品| 国产原创中文字幕| 欧美激情一区二区三区视频高清| 高清一级做a爱过程不卡视频| 二级片在线观看| 成人免费一级纶理片| 国产成人精品综合| 99久久精品国产国产毛片 | 国产一级强片在线观看| 国产一区免费在线观看| 日本伦理片网站| 国产不卡在线播放| 成人免费福利片在线观看| 日韩在线观看网站| 精品久久久久久综合网| 日本特黄特色aa大片免费| 欧美另类videosbestsex高清| 国产一区免费观看| 国产成人啪精品| 精品久久久久久中文字幕一区| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 91麻豆精品国产综合久久久| 四虎影视精品永久免费网站| 欧美一级视频免费| 久久成人亚洲| 亚洲精品中文字幕久久久久久| 国产91精品系列在线观看| 九九热国产视频| 99久久精品国产麻豆| 国产麻豆精品视频| 久久99爰这里有精品国产| 91麻豆tv| 午夜激情视频在线观看| 亚洲 激情| 一级片片|