首頁 > 軟體

golang結合mysql設定最大連線數和最大空閒連線數

2022-02-20 13:00:42

本文介紹golang 中連線MySQL時,如何設定最大連線數和最大空閒連線數。

關於最大連線數和最大空閒連線數,是定義在golang標準庫中database/sql的。

文中例子連線MySQL用的SQL driver package是github.com/go-sql-driver/mysql.

設定最大連線數的介面是

func (db *DB) SetMaxOpenConns(n int) 

設定連線MySQL可以開啟的最大連線數。

如果n <= 0,表示開啟的連線數沒有限制。

預設為0,也就是不限制連線數。

另一個與連線數相關的引數是MaxIdleConns,表示最大空閒連線數。

如果MaxIdleConns 大於0,且大於MaxOpenConns,那麼將調整MaxIdleConns等於MaxOpenConns,有多餘的連線時會關閉多餘的連線。

設定最大空閒連線數的介面是:

func (db *DB) SetMaxIdleConns(n int)

如果n<=0,表示不使用空閒連線池,即一個連線如果不使用,不會放入空閒連線池。因此,這種方式不會複用連線,每次執行SQL語句,都會重新建立新的連線。

預設的最大空閒連線數為2:
const defaultMaxIdleConns = 2

關於開啟的連線和空閒的連線之間的關係,補充一下:

開啟的連線 = 正在使用的連線(inuse) + 處於空閒狀態的連線(idle)

下面對最大連線數和最大空閒連線數做下測試和驗證。

1.最大連線數測試

首先設定最大開啟的連線數為1,接著開啟20個goroutine,每個goroutine執行sql語句,列印執行sql使用的連線的connection id。並執行耗時的sql語句佔用連線,觀察其他需要執行SQL的goroutine的執行情況。

例子程式碼如下:

package main

import (
        "database/sql"
        "log"

        _ "github.com/go-sql-driver/mysql"
)

var DB *sql.DB
var dataBase = "root:Aa123456@tcp(127.0.0.1:3306)/?loc=Local&parseTime=true"

func Init() {
        var err error
        DB, err = sql.Open("mysql", dataBase)
        if err != nil {
                log.Fatalln("open db fail:", err)
        }

        DB.SetMaxOpenConns(1)

        err = DB.Ping()
        if err != nil {
                log.Fatalln("ping db fail:", err)
        }
}

func main() {
        Init()
        
        //開啟20個goroutine
        for i:=0; i < 20; i++ {
                go one_worker(i)
        }
        
        select {
        }

}

func one_worker(i int) {
        var connection_id int
        err := DB.QueryRow("select CONNECTION_ID()").Scan(&connection_id)
        if err != nil {
                log.Println("query connection id failed:", err)
                return
        }

        log.Println("worker:", i, ", connection id:", connection_id)

        var result int
        err = DB.QueryRow("select sleep(10)").Scan(&result)
        if err != nil {
                log.Println("query sleep connection id faild:", err)
                return
        }

}

output

2019/10/02 18:14:25 worker: 2 , connection id: 55
2019/10/02 18:14:25 worker: 17 , connection id: 55
2019/10/02 18:14:25 worker: 11 , connection id: 55
2019/10/02 18:14:35 worker: 3 , connection id: 55
2019/10/02 18:14:45 worker: 0 , connection id: 55
2019/10/02 18:14:45 worker: 4 , connection id: 55
2019/10/02 18:14:45 worker: 5 , connection id: 55
2019/10/02 18:15:05 worker: 7 , connection id: 55
2019/10/02 18:15:25 worker: 15 , connection id: 55
2019/10/02 18:15:25 worker: 6 , connection id: 55
2019/10/02 18:15:35 worker: 13 , connection id: 55
2019/10/02 18:15:45 worker: 19 , connection id: 55
2019/10/02 18:15:45 worker: 10 , connection id: 55
2019/10/02 18:15:45 worker: 12 , connection id: 55
2019/10/02 18:15:55 worker: 14 , connection id: 55
2019/10/02 18:16:15 worker: 8 , connection id: 55
2019/10/02 18:16:35 worker: 18 , connection id: 55
2019/10/02 18:16:35 worker: 1 , connection id: 55
2019/10/02 18:17:05 worker: 16 , connection id: 55
2019/10/02 18:17:35 worker: 9 , connection id: 55

使用show processlist檢視連線

mysql> show processlist;
+----+------+-----------------+------+---------+------+------------+------------------+
| Id | User | Host            | db   | Command | Time | State      | Info             |
+----+------+-----------------+------+---------+------+------------+------------------+
| 20 | root | localhost       | NULL | Query   |    0 | starting   | show processlist |
| 55 | root | localhost:59518 | NULL | Query   |    5 | User sleep | select sleep(10) |
+----+------+-----------------+------+---------+------+------------+------------------+
2 rows in set (0.00 sec)

使用netstat 檢視連線

netstat -an | grep 3306
tcp4       0      0  127.0.0.1.3306         127.0.0.1.59518        ESTABLISHED
tcp4       0      0  127.0.0.1.59518        127.0.0.1.3306         ESTABLISHED
tcp46      0      0  *.3306                 *.*                    LISTEN

從結果可以看到,20個goroutine輪流使用同一個連線(connection id 為55)執行sql語句。

當連線被佔用時,其他嘗試使用連線的goroutine會被阻塞。直到連線使用完後,其他goroutine才可以使用連線。

即使多個goroutine在執行SQL,也沒有建立多個連線。

因此,最大連線數設定生效。

有些讀者可能會問,沒有看到設定最大空閒連線數,此時最大空間連線數是多少?

前面已經提到,預設的最大空閒連線數是2.

下面再來測試下最大空間連線數。

2.最大空閒連線數測試

下面例子中,設定最大連線數為1,最大空閒連線數為0.

並且每隔3s執行一條SQL語句。

程式碼如下:

package main

import (
        "database/sql"
        "log"
        "time"

        _ "github.com/go-sql-driver/mysql"

)

var DB *sql.DB
var dataBase = "root:Aa123456@tcp(127.0.0.1:3306)/?loc=Local&parseTime=true"

func mysqlInit() {
        var err error
        DB, err = sql.Open("mysql", dataBase)
        if err != nil {
                log.Fatalln("open db fail:", err)
        }

        DB.SetMaxOpenConns(1)
        DB.SetMaxIdleConns(0)

        err = DB.Ping()
        if err != nil {
                log.Fatalln("ping db fail:", err)
        }
}

func main() {
        mysqlInit()

        for {
                execSql()
                time.Sleep(3*time.Second)
        }
}


func execSql() {
        var connection_id int
        err := DB.QueryRow("select CONNECTION_ID()").Scan(&connection_id)
        if err != nil {
                log.Println("query connection id failed:", err)
                return
        }

        log.Println("connection id:", connection_id)
}

output:

2019/10/13 23:06:00 connection id: 26
2019/10/13 23:06:03 connection id: 27
2019/10/13 23:06:06 connection id: 28
2019/10/13 23:06:09 connection id: 29
2019/10/13 23:06:12 connection id: 30
2019/10/13 23:06:15 connection id: 31
2019/10/13 23:06:18 connection id: 32
2019/10/13 23:06:21 connection id: 33
2019/10/13 23:06:24 connection id: 34
2019/10/13 23:06:27 connection id: 35
2019/10/13 23:06:30 connection id: 36
2019/10/13 23:06:33 connection id: 37
2019/10/13 23:06:36 connection id: 38

從結果中可以看出,每次執行SQL使用的連線connection id都不同。

設定最大空閒連線數為0,每次執行SQL後,連線不會放入空閒連線池,而是會被關閉,下次執行SQL時,會重新建立新的連線。

3.參考

Connection pool and timeouts

Golang MySQL 連線和連線池

到此這篇關於golang結合mysql設定最大連線數和最大空閒連線數的文章就介紹到這了,更多相關golang 最大連線數和最大空閒連線數內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com