天天看點

Go與Redis連接配接池的那些事兒~

Go與Redis連接配接池的那些事兒~

一、什麼是連接配接池,連接配接池有什麼用

先看看别人是怎麼介紹連接配接池的吧:

連接配接池基本的思想是在系統初始化的時候,将資料庫連接配接作為對象存儲在記憶體中,當使用者需要通路資料庫時,并非建立一個新的連接配接,而是從連接配接池中取出一個已建立的空閑連接配接對象。使用完畢後,使用者也并非将連接配接關閉,而是将連接配接放回連接配接池中,以供下一個請求通路使用。而連接配接的建立、斷開都由連接配接池自身來管理。同時,還可以通過設定連接配接池的參數來控制連接配接池中的初始連接配接數、連接配接的上下限數以及每個連接配接的最大使用次數、最大空閑時間等等。也可以通過其自身的管理機制來監視資料庫連接配接的數量、使用情況等。

下面我來簡單解釋一下,因為每次

Redis

用戶端連接配接

Redis

服務端都需要一段時間,而處理各種操作的時間很多時候都很短,如果每次進行各種操作時都需要重新連接配接

Redis

,那麼就會浪費大量時間。是以

Redis

引入連接配接池,連接配接池可以實作建立多個用戶端連接配接而不釋放,避免浪費IO資源,不使用的時候就放在連接配接池,這樣就減少了連接配接資料庫所需要的時間,提高效率。

連接配接池就是建一個池子和一定量的管道。每次當管道被取盡時,就不能繼續消耗IO資源了,這樣就保證了IO資源不會耗盡。

二、代碼展示

package main

import (
    "fmt"
    "github.com/garyburd/redigo/redis"
    "strconv"
    "time"
)

func main() {
    pool := &redis.Pool{
        // Maximum number of connections allocated by the pool at a given time.
        // When zero, there is no limit on the number of connections in the pool.
        //最大活躍連接配接數,0代表無限
        MaxActive: 888,
        //最大閑置連接配接數
        // Maximum number of idle connections in the pool.
        MaxIdle: 20,
        //閑置連接配接的逾時時間
        // Close connections after remaining idle for this duration. If the value
        // is zero, then idle connections are not closed. Applications should set
        // the timeout to a value less than the server's timeout.
        IdleTimeout: time.Second * 100,
        //定義撥号獲得連接配接的函數
        // Dial is an application supplied function for creating and configuring a
        // connection.
        //
        // The connection returned from Dial must not be in a special state
        // (subscribed to pubsub channel, transaction started, ...).
        Dial: func() (redis.Conn, error) {
            return redis.Dial("tcp","127.0.0.1:6379"),
        }
    }
    //延遲關閉連接配接池
    defer pool.Close()
    //IO并發連接配接
    for i:=0;i<10;i++{
        go getConnFromPoolAndHappy(pool,i)
    }
    //保持主協程存活
    time.Sleep(3*time.Second)

}

func getConnFromPoolAndHappy(pool *redis.Pool, i int)  {
    //通過連接配接池獲得連接配接
    conn := pool.Get()
    //延時關閉連接配接
    defer conn.Close()
    //使用連接配接操作資料
    reply, err := conn.Do("set", "conn"+strconv.Itoa(i), i)
    s, _ := redis.String(reply, err)
    fmt.Println(s)
}           
MaxActive: 888           

上面的代碼表示在給定的時間内被連接配接池配置設定的最大連接配接數,當該值為0時,表示連接配接池的連接配接數是無上限的。

MaxIdle: 20           

表示連接配接池閑置連接配接數的上限。

IdleTimeout: time.Second * 100           

該值表示如果連接配接池的的連接配接閑置超過該值就會關閉連接配接。如果該值為零,連接配接池中閑置的連接配接就不會關閉。應用程式應該設定這個限制逾時時間不超過服務端的限制逾時時間。

Dial: func() (redis.Conn, error) {
   return redis.Dial("tcp", "127.0.0.1:6379")
}           

該值為一個函數,該函數應該用于創造和配置連接配接。從Dial傳回的連接配接不能處于特殊狀态。

開辟一條并發協程執行該函數,因為for循環的範圍是0-9,是以并發數為10,表示同時有10個人去連接配接池取管道。

go getConnFromPoolAndHappy(pool,i)           

下列代碼是用來保持主協程存活的。

time.Sleep(3*time.Second)           

下面表示通過連接配接池獲得連接配接。也就是從池子拿一根管道。

conn := pool.Get()           

然後下面的代碼是使用管道。

reply, err := conn.Do("set", "conn"+strconv.Itoa(i), i)    s, _ := redis.String(reply, err)    fmt.Println(s)           

如果一個人占着管道不用,就會被閑置。如果在閑置處太久不動,達到閑置連接配接的逾時時間,就會被請走。

因為池子使用的都是同一個,是以需要使用池子的指針。

pool := &redis.Pool