Python中使用Redis遭遇Too many open files

今天服务突然爆出好多异常日志:
ConnectionError: Error 24 connecting to 127.0.0.1:6379. Too many open files.

通过命令cat /proc/(redis-sever_PID)/limits获得

Max open files  4016  4016  files

第一反应是觉得可能4016过小,所以需要对其升值。

编辑/etc/sysctl.conf 添加如下内容:

fs.file-max = 65535

运行sysctl -p

再编辑 /etc/security/limits.conf:

root soft nofile 65536
root hard nofile 65536
* soft nofile 65536
* hard nofile 65536

使用命令ulimit -n 65535,将最大文件数提高至65535,这种修改只是在当前shell中生效,想要真正对Redis中Max open files进行修改,就需要在当前shell中进行重启redis,然后服务就恢复了正常。这种方法其实是治标不治本的

在临时解决异常问题后,需要思考为什么会有这么多未释放的连接。

经过分析,在创建Redis连接时,都会创建一个ConnectionPool。这导致,每获取一个StrictRedis时,都会创建一个新的ConnectionPool,这才是导致文件数过多的原因。

解决方案:

先创建创建一个ConnectionPool
pool = redis.ConnectionPool(host='localhost', port=6379, db=0)

然后创建一个Redis
r = redis.Redis(connection_pool=pool)

修改前的代码

import os
import logging

import redis

logger = logging.getLogger(__name__)


def redis_connect(db=2):
    """
    Established an resid connection.
    """
    REDIS_HOST = '127.0.0.1'
    REDIS_PORT = 6379
    REDIS_PWD = None
    REDIS_USER = None

    redis_client = redis.Redis(host=REDIS_HOST, port=int(
        REDIS_PORT), db=db, password=REDIS_PWD, retry_on_timeout=True)

    return redis_client

修改后的代码

# coding: utf-8

import os
import logging

import redis

redistogo_url = os.getenv('REDISTOGO_URL', None)
if not redistogo_url:
    REDIS_HOST = '127.0.0.1'
    REDIS_PORT = 6379
    REDIS_PWD = None
    REDIS_USER = None
else:
    redis_url = redistogo_url
    redis_url = redis_url.split('redis://')[1]
    redis_url = redis_url.split('/')[0]
    REDIS_USER, redis_url = redis_url.split(':', 1)
    REDIS_PWD, redis_url = redis_url.split('@', 1)
    REDIS_HOST, REDIS_PORT = redis_url.split(':', 1)

pool_2 = redis.ConnectionPool(host=REDIS_HOST, port=int(
    REDIS_PORT), db=2, retry_on_timeout=True)
pool_3 = redis.ConnectionPool(host=REDIS_HOST, port=int(
    REDIS_PORT), db=3, retry_on_timeout=True)


def redis_connect(db=2):
    """
    Established an resid connection.
    """
    if db == 2:
        redis_client = redis.Redis(connection_pool=pool_2)
    if db == 3:
        redis_client = redis.Redis(connection_pool=pool_3)
    return redis_client
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 回顾问题 在《追踪Redis Sentinel的CPU占有率长期接近100%的问题》[https://www.ji...
    宅楠军阅读 4,431评论 3 5
  • 安全性 设置客户端连接后进行任何其他指令前需要使用的密码。 警告:因为redis 速度相当快,所以在一台比较好的服...
    OzanShareing阅读 1,824评论 1 7
  • 1 Redis介绍1.1 什么是NoSql为了解决高并发、高可扩展、高可用、大数据存储问题而产生的数据库解决方...
    克鲁德李阅读 5,354评论 0 36
  • 转山,转水,转几世宝塔,转万面经幡。驾梦南行的你,与我在途中相遇,又与我在途中相隔万里……
    绛伶人阅读 221评论 0 0
  • 昨晚,卢打电话过来,说,猫腻在于16的教绩真的有假。 还有现在的17名也有假。 若这两人下,就是她和18并列第16...
    哆啦__阅读 225评论 0 0