校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > golang實現基于channel的通用連接池詳解

golang實現基于channel的通用連接池詳解

熱門標簽:釘釘有地圖標注功能嗎 建造者2地圖標注 鄭州亮點科技用的什么外呼系統 黃岡人工智能電銷機器人哪個好 阿里云ai電話機器人 浙江高頻外呼系統多少錢一個月 汕頭小型外呼系統 濱州自動電銷機器人排名 惠州電銷防封電話卡

前言

golang的channel除了goroutine通信之外還有很多其他的功能,本文將實現一種基于channel的通用連接池。下面話不多說了,來一起看看詳細的介紹吧。

功能

* 連接池中連接類型為interface{},使得更加通用

* 鏈接的最大空閑時間,超時的鏈接將關閉丟棄,可避免空閑時鏈接自動失效問題

* 使用channel處理池中的鏈接,高效

何為通用?

連接池的實現不依賴具體的實例,而依賴某個接口,本文的連接池選用的是io.Closer接口,只要是實現了該接口的對象都可以被池管理。

當然,你可以實現基于interface{}的連接池,這樣任何對象都可以被管理。

實現原理

將連接句柄存入channel中,由于緩存channel的特性,獲取連接時如果池中有連接,將直接返回,如果池中沒有連接,將阻塞或者新建連接(沒超過最大限制的情況下)。

由于面向接口編程,所有創建連接的邏輯是不清楚的,這里需要傳入一個函數,該函數返回一個io.Closer對象。

實現

由于并發問題,在需要操作池中互斥數據的時候需要加鎖。

package pool
import (
  "errors"
  "io"
  "sync"
  "time"
)

var (
  ErrInvalidConfig = errors.New("invalid pool config")
  ErrPoolClosed  = errors.New("pool closed")
)

type factory func() (io.Closer, error)

type Pool interface {
  Acquire() (io.Closer, error) // 獲取資源
  Release(io.Closer) error   // 釋放資源
  Close(io.Closer) error    // 關閉資源
  Shutdown() error       // 關閉池
}

type GenericPool struct {
  sync.Mutex
  pool    chan io.Closer
  maxOpen   int // 池中最大資源數
  numOpen   int // 當前池中資源數
  minOpen   int // 池中最少資源數
  closed   bool // 池是否已關閉
  maxLifetime time.Duration
  factory   factory // 創建連接的方法
}

func NewGenericPool(minOpen, maxOpen int, maxLifetime time.Duration, factory factory) (*GenericPool, error) {
  if maxOpen = 0 || minOpen > maxOpen {
    return nil, ErrInvalidConfig
  }
  p := GenericPool{
    maxOpen:   maxOpen,
    minOpen:   minOpen,
    maxLifetime: maxLifetime,
    factory:   factory,
    pool:    make(chan io.Closer, maxOpen),
  }

  for i := 0; i  minOpen; i++ {
    closer, err := factory()
    if err != nil {
      continue
    }
    p.numOpen++
    p.pool - closer
  }
  return p, nil
}

func (p *GenericPool) Acquire() (io.Closer, error) {
  if p.closed {
    return nil, ErrPoolClosed
  }
  for {
    closer, err := p.getOrCreate()
    if err != nil {
      return nil, err
    }
    // todo maxLifttime處理
    return closer, nil
  }
}

func (p *GenericPool) getOrCreate() (io.Closer, error) {
  select {
  case closer := -p.pool:
    return closer, nil
  default:
  }
  p.Lock()
  if p.numOpen >= p.maxOpen {
    closer := -p.pool
    p.Unlock()
    return closer, nil
  }
  // 新建連接
  closer, err := p.factory()
  if err != nil {
    p.Unlock()
    return nil, err
  }
  p.numOpen++
  p.Unlock()
  return closer, nil
}

// 釋放單個資源到連接池
func (p *GenericPool) Release(closer io.Closer) error {
  if p.closed {
    return ErrPoolClosed
  }
  p.Lock()
  p.pool - closer
  p.Unlock()
  return nil
}

// 關閉單個資源
func (p *GenericPool) Close(closer io.Closer) error {
  p.Lock()
  closer.Close()
  p.numOpen--
  p.Unlock()
  return nil
}

// 關閉連接池,釋放所有資源
func (p *GenericPool) Shutdown() error {
  if p.closed {
    return ErrPoolClosed
  }
  p.Lock()
  close(p.pool)
  for closer := range p.pool {
    closer.Close()
    p.numOpen--
  }
  p.closed = true
  p.Unlock()
  return nil
}

結論

基于該連接池,可以管理所有io.Closer對象。比如memcached,redis等等,非常方便!

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • 基于golang channel實現的輕量級異步任務分發器示例代碼
  • golang中for循環遍歷channel時需要注意的問題詳解
  • Golang優雅關閉channel的方法示例
  • golang中單向channel的語法介紹
  • golang判斷chan channel是否關閉的方法
  • Golang中channel使用的一些小技巧
  • Golang中channel的原理解讀(推薦)

標簽:東營 瀘州 阿壩 晉中 昭通 泰安 駐馬店 滄州

巨人網絡通訊聲明:本文標題《golang實現基于channel的通用連接池詳解》,本文關鍵詞  golang,實現,基于,channel,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《golang實現基于channel的通用連接池詳解》相關的同類信息!
  • 本頁收集關于golang實現基于channel的通用連接池詳解的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 阿拉尔市| 武鸣县| 中牟县| 措勤县| 黑河市| 香格里拉县| 乌拉特中旗| 玉屏| 慈溪市| 鲁山县| 射阳县| 东乡县| 南昌市| 迁安市| 老河口市| 锡林浩特市| 阜南县| 柞水县| 临汾市| 金川县| 鹤岗市| 酉阳| 岑溪市| 延庆县| 莆田市| 龙州县| 伊宁市| 商南县| 阜平县| 大宁县| 广汉市| 德安县| 荃湾区| 潼关县| 毕节市| 公主岭市| 丰原市| 筠连县| 广汉市| 招远市| 德兴市|