php redis断线重连

15 篇文章 1 订阅
1 篇文章 0 订阅
php redis断线重连,pconnect连接失败问题

介绍

在swoole ,workerman等cli长连接模式下,遇到Redis异常断开,后面又开启的情况,一般得重新启动程序才能正常使用,

本文介绍在不重启服务,实现原来的Redis断线重连

 

原理

Redis 断开的情况下调用

$Redis->ping()会触发Notice错误,Notice: Redis::ping(): send of 14 bytes failed with errno=10054

当获取redis实例时,如果ping不通或者出现异常,就重新连接

实现1

因为try catch  捕捉不到notice异常,所以ping不通直接重新连接,catch捕捉新连接的实例没有连接上,下次执行ping触发

Redis server went away 异常
    public static function getInstance( )
    {
        try {
            if (!self::$_instance) {
                new self();
            } else {
                if (!self::$_instance->ping())
                    new self();
            }
        } catch (\Exception $e) {
            // 断线重连
            new self();
        }
        return self::$_instance;
    }

实现2

1调用ping之前先抛出个notice异常,

2调用ping

3用error_get_last获取最后一个错误,如果错误信息跟我们抛出的一样,说明ping通了,否则抛出个异常 ,让catch捕捉到执行重连,

当重连一次没连上再次调用$_instance->ping()会直接抛出Redis server went away异常让catch捕捉到

    public static function getInstance( )
    {
        if (!self::$_instance) {
            new self();
        }
        else{
            try {
                @trigger_error('flag', E_USER_NOTICE);
                self::$_instance->ping();
                $error = error_get_last();
                if($error['message'] != 'flag')
                    throw new \Exception('Redis server went away');
            } catch (\Exception $e) {
                // 断线重连
                new self();
            }
        }
        return self::$_instance;
    }

 

Redis类完整代码

<?php


namespace lib;


class Redis
{

    private static $_instance; //存储对象
    private function __construct( ){
        $config = Config::get('redis');
        self::$_instance = new \Redis();
        //从配置读取
        self::$_instance->pconnect($config['host'], $config['port']);
        if ('' != $config['password']) {
            self::$_instance->auth($config['password']);
        }

    }




    public static function getInstance( )
    {
        if (!self::$_instance) {
            new self();
        }
        else{
            try {
                @trigger_error('flag', E_USER_NOTICE);
                self::$_instance->ping();
                $error = error_get_last();
                if($error['message'] != 'flag')
                    throw new \Exception('Redis server went away');
            } catch (\Exception $e) {
                // 断线重连
                new self();
            }
        }
        return self::$_instance;
    }

//    public static function getInstance( )
//    {
//        try {
//            if (!self::$_instance) {
//                new self();
//            } else {
//                if (!self::$_instance->ping())
//                    new self();
//            }
//        } catch (\Exception $e) {
//            // 断线重连
//            new self();
//        }
//        return self::$_instance;
//    }



    /**
    * 禁止clone
    */
    private function __clone(){}

    /**
     * 其他方法自动调用
     * @param $method
     * @param $args
     * @return mixed
     */
    public function __call($method,$args)
    {
        return call_user_func_array([self::$_instance, $method], $args);
    }

    /**
     * 静态调用
     * @param $method
     * @param $args
     * @return mixed
     */
    public static function __callStatic($method,$args)
    {
        self::getInstance();
        return call_user_func_array([self::$_instance, $method], $args);
    }



}

调用

$this->handler = Redis::getInstance();
        $key    = $this->getCacheKey($name);
        $value = $this->handler->get($key);

补充

pconnect建立连接后重连失败问题

经测试长连接下使用pconnect建立连接后,redis超时被动断开了链接,

$res = self::$_instance->pconnect($config['host'], $config['port']); 

$res 会返回true,但不是新建的链接,调用$res-get()会报错

原因

研究发现

使用pconnect,链接在php进程的整个生命周期内被重用, close的作用仅是使当前php不能再进行redis请求,但无法真正关闭redis长连接,连接在后续请求中仍然会被重用,直至fpm进程生命周期结束。

长连接中只有进程被停止,连接才会断开,所以连接断开时new不起作用,返回连接成功,而事实上已经断了,还是最早的那个连接,从而导致不能进行后续读取数据操作

所以长连接中请使用connect,

有帮到你的话请三连哦

在 Spring Boot 中使用 Redis 的过程中,如果 Redis 服务器由于某些原因宕机或网络中断,那么就需要重新连接 Redis 服务器。以下是在 Spring Boot 中实现 Redis 断线重连的步骤: 1. 在 application.properties 或 application.yml 文件中配置 Redis 连接参数,例如: ``` spring.redis.host=127.0.0.1 spring.redis.port=6379 spring.redis.password= spring.redis.database=0 ``` 2. 创建 RedisConnectionFactory 实例,并设置 Redis 连接参数。可以使用默认的 JedisConnectionFactory 或自定义的 LettuceConnectionFactory。例如: ``` @Configuration public class RedisConfig { @Value("${spring.redis.host}") private String host; @Value("${spring.redis.port}") private int port; @Value("${spring.redis.password}") private String password; @Value("${spring.redis.database}") private int database; @Bean public RedisConnectionFactory redisConnectionFactory() { LettuceConnectionFactory lettuceConnectionFactory = new LettuceConnectionFactory(); lettuceConnectionFactory.setHostName(host); lettuceConnectionFactory.setPort(port); lettuceConnectionFactory.setPassword(password); lettuceConnectionFactory.setDatabase(database); lettuceConnectionFactory.afterPropertiesSet(); return lettuceConnectionFactory; } } ``` 3. 创建 RedisTemplate 实例,并设置 RedisConnectionFactory。例如: ``` @Configuration public class RedisConfig { ... @Bean public RedisTemplate<String, Object> redisTemplate() { RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(redisConnectionFactory()); redisTemplate.setKeySerializer(new StringRedisSerializer()); redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer()); redisTemplate.afterPropertiesSet(); return redisTemplate; } } ``` 4. 在 RedisConnectionFactory 上添加 ConnectionListener 监听器,用于监听 Redis 连接状态变化。例如: ``` @Configuration public class RedisConfig { ... @Bean public RedisConnectionFactory redisConnectionFactory() { LettuceConnectionFactory lettuceConnectionFactory = new LettuceConnectionFactory(); lettuceConnectionFactory.setHostName(host); lettuceConnectionFactory.setPort(port); lettuceConnectionFactory.setPassword(password); lettuceConnectionFactory.setDatabase(database); lettuceConnectionFactory.afterPropertiesSet(); lettuceConnectionFactory.addConnectionListener(new RedisConnectionListener()); return lettuceConnectionFactory; } private static class RedisConnectionListener implements RedisConnectionFailureListener { private final Logger logger = LoggerFactory.getLogger(RedisConnectionListener.class); @Override public void onMasterFailure(RedisException e) { logger.error("Redis master failure", e); } @Override public void onSlaveFailure(RedisException e) { logger.error("Redis slave failure", e); } @Override public void onRefresh(RedisConnection connection) { logger.info("Redis connection refresh"); } } } ``` 在 RedisConnectionFailureListener 中,可以根据不同的 RedisException 进行不同的处理。例如,当 RedisException 是连接异常时,可以进行重连操作,例如: ``` @Override public void onMasterFailure(RedisException e) { logger.error("Redis master failure", e); if (e instanceof RedisConnectionException) { redisConnectionFactory.resetConnection(); } } ``` 这样,在 Redis 服务器宕机或网络中断时,RedisConnectionFactory 就会自动进行重连操作。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

flysnownet

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值