溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang中怎么利用channel 實現一個連接池

發布時間:2021-07-06 15:13:30 來源:億速云 閱讀:278 作者:Leah 欄目:大數據

這篇文章給大家介紹golang中怎么利用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
}

關于golang中怎么利用channel 實現一個連接池就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女