Go操作Redis封装的方法

测试项目目录

项目代码

operates.go

package redisOperate

import (
    "errors"
    "github.com/garyburd/redigo/redis"
    "math"
    "time"
)


type RRedis struct {
    redisCli       *redis.Pool
    maxIdle        int
    maxActive      int
    maxIdleTimeout time.Duration
    maxTimeout     time.Duration
    lazyLimit      bool
    maxSize        int
}

// 从池里获取连接 ———— 单独的方法
func (r *RRedis) getRedisConn() redis.Conn {
    rc := r.redisCli.Get()
    // // 用完后将连接放回连接池
    // defer rc.Close()
    return rc
}

// GetAllKeys 获取所有keys
func (r *RRedis) GetAllKeys() []string {

    rc := r.getRedisConn()
    defer rc.Close()

    keys, err := redis.Strings(rc.Do("KEYS", "*"))
    if err != nil {
        return make([]string, 0)
    }
    return keys
}

func (r *RRedis) Get(key string, timeout int) (string, error) {

    start := time.Now()

    for {
        res, err := r.GetNoWait(key)
        if err != nil {
            return "", err
        } else if res == "" {
            if timeout != -1 {
                lasted := time.Now().Sub(start)
                if r.maxTimeout > lasted {
                    t1 := r.maxTimeout
                    t2 := time.Duration(timeout)*time.Second - lasted
                    time.Sleep(time.Duration(math.Min(float64(t1), float64(t2))))
                } else {
                    return "", errors.New("GET timeout")
                }
            } else {
                time.Sleep(r.maxTimeout)
            }
        } else {
            return res, nil
        }
    }
}

func (r *RRedis) GetNoWait(key string) (string, error) {

    rc := r.getRedisConn()
    defer rc.Close()

    res, err := redis.String(rc.Do("LPOP", key))

    if err != nil {
        return "", err
    }
    return res, nil
}

func (r *RRedis) Put(key string, value string, timeout int) (int, error) {

    start := time.Now()

    for {
        res, err := r.PutNoWait(key, value)

        if err != nil {
            return 0, err
        } else if res == -1 {
            if timeout != -1 {
                lasted := time.Now().Sub(start)
                if r.maxTimeout > lasted {
                    t1 := r.maxTimeout
                    t2 := time.Duration(timeout)*time.Second - lasted
                    time.Sleep(time.Duration(math.Min(float64(t1), float64(t2))))
                } else {
                    return 0, errors.New("PUT timeout")
                }
            } else {
                time.Sleep(r.maxTimeout)
            }

        } else {
            return res, nil
        }

    }
}

func (r *RRedis) PutNoWait(key string, value string) (int, error) {

    rc := r.getRedisConn()
    defer rc.Close()

    if r.Full(key) {
        return -1, nil
    }

    res, err := redis.Int(rc.Do("RPUSH", key, value))
    if err != nil {
        return 0, err
    }
    return res, nil
}

func (r *RRedis) QSize(key string) int {

    rc := r.getRedisConn()
    defer rc.Close()

    res, err := redis.Int(rc.Do("LLEN", key))
    if err != nil {
        return -1
    }

    return res
}

func (r *RRedis) Empty(key string) bool {

    rc := r.getRedisConn()
    defer rc.Close()

    res, err := redis.Int(rc.Do("LLEN", key))
    if err != nil {
        return false
    }
    if res == 0 {
        return true
    }
    return false
}

func (r *RRedis) Full(key string) bool {

    if r.maxSize != 0 && r.QSize(key) >= r.maxSize {
        return true
    }

    return false
}
operates.go

redisInter.go

package redisOperate

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

// 父级interface
type RedisFatherInterface interface {
    // 获取所有keys
    GetAllKeys() []string
}



// ListInterface 操作list接口
type ListInterface interface {
    // "继承"父类的所有方法
    RedisFatherInterface
    GetNoWait(key string) (string, error) // ~di: interface{}类型、数据压缩
    Get(key string, timeout int) (string, error)
    Put(key string, value string, timeout int) (int, error)
    PutNoWait(key string, value string) (int, error)
    QSize(key string) int
    Empty(key string) bool
    Full(key string) bool
}


// redis所有操作的接口
type RedisInterface interface {
    ListInterface
}


// 工厂函数,要求对应的结构体必须实现 RedisInterface 中的所有方法
// 如果只想实现某一些方法,就返回"有这些方法的结构体"就好了
func ProduceRedis(host, port, password string, db, maxSize int, lazyLimit bool) (RedisInterface, error) {
    
    // 要求RRedis结构体实现返回的接口中所有的方法!
    redisObj := &RRedis{
        maxIdle:        100,
        maxActive:      130,
        maxIdleTimeout: time.Duration(60) * time.Second,
        maxTimeout:     time.Duration(30) * time.Second,
        lazyLimit:      lazyLimit,
        maxSize:        maxSize,
    }
    // 建立连接池
    redisObj.redisCli = &redis.Pool{
        MaxIdle:     redisObj.maxIdle,
        MaxActive:   redisObj.maxActive,
        IdleTimeout: redisObj.maxIdleTimeout,
        Wait:        true,
        Dial: func() (redis.Conn, error) {
            con, err := redis.Dial(
                "tcp",
                host+":"+port, // address
                redis.DialPassword(password),
                redis.DialDatabase(int(db)),
                redis.DialConnectTimeout(redisObj.maxTimeout),
                redis.DialReadTimeout(redisObj.maxTimeout),
                redis.DialWriteTimeout(redisObj.maxTimeout),
            )
            if err != nil {
                return nil, err
            }
            return con, nil
        },
    }

    return redisObj, nil
}

main.go

package main

import (
    "fmt"
    red "ginProjects/redisOperate"
)

func main() {
// 测试redis的操作 testRedisOperate() } // 测试redis的操作 func testRedisOperate(){ redisCLi, err := red.ProduceRedis("127.0.0.1","6388","",0,100,true) if err != nil{ fmt.Println("redis连接错误!err>>>",err.Error()) return } // 获取所有keys allKeysLst := redisCLi.GetAllKeys() fmt.Print("key>>> ",allKeysLst) }

~~~

posted on 2020-10-28 11:46  江湖乄夜雨  阅读(1869)  评论(1编辑  收藏  举报