【发布时间】:2015-01-16 00:31:26
【问题描述】:
我正在尝试将 2 亿个密钥加载到 redis 中,并且通常在大约 3100 万个密钥处开始出现错误并且不得不停止。我正在使用 golang 和 redis 库“github.com/garyburd/redigo/redis”
我是这样设置连接池的:
func newPool(server string) *redis.Pool {
return &redis.Pool{
MaxIdle: 3,
MaxActive: 10,
IdleTimeout: 240 * time.Second,
Dial: func () (redis.Conn, error) {
c, err := redis.Dial("tcp", server)
if err != nil {
return nil, err
}
return c, err
},
TestOnBorrow: func(c redis.Conn, t time.Time) error {
_, err := c.Do("PING")
return err
},
}
}
然后我尝试用这个函数填充 redis:
func RedisServerBatchLoadKeys(rtbExchange string, keys []string){
redisLock.Lock()
defer redisLock.Unlock()
retry := 0
for {
conn := GetConnOrPanic(rtbExchange)
defer conn.Close()
conn.Send("MULTI")
for _, key := range keys {
conn.Send("SET", key, maxCount)
conn.Send("EXPIRE", key, numSecondsExpire)
}
_, err := conn.Do("EXEC")
if err == nil {
break
} else if !(err == io.EOF) {
CheckRedisError(err, rtbExchange, "Could not load batch")
} else {
retry ++
}
if retry >= 10 {
CheckRedisError(err, rtbExchange, "Could not load batch - 10 retries")
}
}
}
我遇到了很多错误,例如:
- 读取 tcp 10.249.15.194:6379:对等方重置连接
- 拨号 tcp 10.249.15.194:6379:连接被拒绝
- redis#RedisError : EOF
我是在做一些根本错误的事情,还是我必须添加更多错误检查(除了我添加的 EOF)。
谢谢,
【问题讨论】:
-
我不知道
GetConnOrPanic()是做什么的,但是您是否想在一次交易中设置数百万个值?将每个 SET/EXPIRE 放入单个 MULTI/EXEC 事务中。 -
抱歉,我一次设置了 100 个
-
您似乎在每次迭代时都从池中获得了一个新连接,而没有关闭它。 defer 只在函数退出时关闭它。
-
我建议删除外部重试循环。如果应用程序未能设置密钥,则说明出现问题。重试可能会掩盖问题。使用 SETEX 使用单个命令设置密钥和过期时间。检查您在单个事务中设置的密钥数量。在事务中设置一百万个密钥将失败。您能否展示 GetConnOrPanic 的代码或解释它的作用?