golang开发中 redis连接池的使用

作为一个phper,第一次听到连接池还有点蒙圈,转golang开发后连接池的概念会经常使用。

一、连接池是什么

连接池是什么?一个服务端资源的连接数量都是有限的,每次初始化时他建一定数量的连接,先把所有连接存起来,谁要用则从里面取,用完后放回去。如果超出连接池容量,要是排队等着或么直接丢弃。

比如我们做开发中常用的mysq,redis,php-fpm的配置
1,redis服务端设置
maxclients 最大连接数设置
2, mysql服务端设置
max_connections 最大连接数
3,PHP-FPM 服务端设置
max_children 最大子进程数
start_servers 起始进程数

我们golang开发时连接redis用到自己设计的连接池概念,想要达到的效果是什么?
1,最大的连接数,最大是为了控制整个系统连接redis服务端的总数。超过个数量不再与reids进行连接。
2,最大的空闲连接数,最大空闲是想在没有redis操作时还可以保持的连接个数,这些连接会一直保持与redis服务端的连接(当然如果redis服务端设置了timeout非0,或是你在代码中手动设置了空闲连接超时间除外)。

二、如何使用Redis连接池

关于go的redis包github里常用的有两个:
github.com/gomodule/redigo/redis
github.com/go-redis/redis

“github.com/go-redis/redis” 这个包里的连接池使用有些模糊,作为一个golang新手不太建议使用这个包来做连接池的使用。

例子1:go-redis/redis 这个包的简单使用
package main
import (
    "fmt"
    "github.com/go-redis/redis"
    "time"
)

func main() {
    var addr = "127.0.0.1:6379"
    var password = ""

    c := redis.NewClient(&redis.Options{
        Addr:     addr,
        Password: password,
    })
    p, err := c.Ping().Result()
    if err != nil {
        fmt.Println("redis kill")
    }
    fmt.Println(p)
    c.Do("SET", "key", "duzhenxun")
    rs := c.Do("GET", "key").Val()
    fmt.Println(rs)
    c.Close()
}

例子2: “github.com/gomodule/redigo/redis” 这个包中有连接池的Api,非常好用
package main
import (
    "fmt"
    redigo "github.com/gomodule/redigo/redis"
    "time"
)

func main() {
    var addr = "127.0.0.1:6379"
    var password = ""

    pool := PoolInitRedis(addr, password)
    c1 := pool.Get()
    c2:=pool.Get()
    c3:=pool.Get()
    c4:=pool.Get()
    c5:=pool.Get()
    fmt.Println(c,c2,c3,c4,c5)
    time.Sleep(time.Second * 5)//redis一共有多少个连接??
    c1.Close()
    c2.Close()
    c3.Close()
    c4.Close()
    c5.Close()
    time.Sleep(time.Second*5) //redis一共有多少个连接??

    //下次是怎么取出来的??
    b1:=pool.Get()
    b2:=pool.Get()
    b3:=pool.Get()
    fmt.Println(b1,b2,b3)
    time.Sleep(time.Second*5)
    b1.Close()
    b2.Close()
    b3.Close()

    //redis目前一共有多少个连接??
    for{
        fmt.Println("主程序运行中....")
        time.Sleep(time.Second*1) 
    }
}

// redis pool
func PoolInitRedis(server string, password string) *redigo.Pool {
    return &redigo.Pool{
        MaxIdle:     2,//空闲数
        IdleTimeout: 240 * time.Second,
        MaxActive:   3,//最大数
        Dial: func() (redigo.Conn, error) {
            c, err := redigo.Dial("tcp", server)
            if err != nil {
                return nil, err
            }
            if password != "" {
                if _, err := c.Do("AUTH", password); err != nil {
                    c.Close()
                    return nil, err
                }
            }
            return c, err
        },
        TestOnBorrow: func(c redigo.Conn, t time.Time) error {
            _, err := c.Do("PING")
            return err
        },
    }
}

三、参数配置说明

MaxIdle:最大空闲连接数,没有redis操作进依然可以保持这个连接数量
MaxActive:最大连接数。同一时间最多有这么多的连接

一般go程序运行时选设置redis连接池的初始化。假如我们设置MaxIdle:2,MaxActive:3时。
连接时:调用pool.Get()时,先从MaxIdle中取出可用连接,如果失败,则看当前设置的MaxActive是否有超出最大数,没有超出则创建一个新的连接。
断开时:调用c.Close() 后,看当前连接数,如果比MaxIdle设置的数量大,则关闭redis连接。反之将连接放回到连接池中。大体如下流程图所示:


图1 获取redis连接
图2 关闭redis连接

四、分析与实验

例子2中,程序启动时休息的5秒里,我们先看一下redis中的当前的连接数,当前连接数只有1个,这个是当前cli的连接。说明go还没有和redis进行连接。我们可以通过info clients 或client list来查看当前redis连接情况。

redis-cli
//当前客户端连接数
127.0.0.1:6379> info clients
# Clients
connected_clients:1

我们get了5次,想是取出5个连接,但我们所设置的MaxActive是3,那redis最大连接数只有3个:

c1 :=pool.Get()
c2:=pool.Get()
c3:=pool.Get()
c4:=pool.Get()
c5:=pool.Get()

可以在redis中使用 info clients 命令查看

127.0.0.1:6379> info clients
# Clients
connected_clients:4 (这里显示4个是因为本身自己的cli连接redis时会有一个连接)

调用关闭接口,发现虽然是关闭了5次。但最终还有2个连接没有关闭。因为我们设置的MaxIdle是2

c1.Close()
c2.Close()
c3.Close()
c4.Close()
c5.Close()

可以在redis中使用 info clients 命令查看

127.0.0.1:6379> info clients
# Clients
connected_clients:3 (这里显示3个是因为本身自己的cli连接redis时会有一个连接)

以上就是我对redis连接池的使用笔记,由于初次尝试使用可能有的地方讲的不对,还请多多指教。微信号:5552123

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容