From a134da6563c14a62f1f9779dfd7fbe8d15a6844e Mon Sep 17 00:00:00 2001 From: thinkphp Date: Thu, 5 May 2016 16:40:02 +0800 Subject: [PATCH] =?UTF-8?q?=E6=94=B9=E8=BF=9Bredisd=E9=A9=B1=E5=8A=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- library/think/cache/driver/Redisd.php | 192 +++++++++++++------------- 1 file changed, 94 insertions(+), 98 deletions(-) diff --git a/library/think/cache/driver/Redisd.php b/library/think/cache/driver/Redisd.php index 4512187d..a11fe810 100644 --- a/library/think/cache/driver/Redisd.php +++ b/library/think/cache/driver/Redisd.php @@ -16,51 +16,51 @@ use think\Exception; use think\Log; /** - 配置参数: - 'cache' => [ - 'type' => 'Redisd' - 'host' => 'A:6379,B:6379', //redis服务器ip,多台用逗号隔开;读写分离开启时,默认写A,当A主挂时,再尝试写B - 'slave' => 'B:6379,C:6379', //redis服务器ip,多台用逗号隔开;读写分离开启时,所有IP随机读,其中一台挂时,尝试读其它节点,可以配置权重 - 'port' => 6379, //默认的端口号 - 'password' => '', //AUTH认证密码,当redis服务直接暴露在外网时推荐 - 'timeout' => 10, //连接超时时间 - 'expire' => false, //默认过期时间,默认0为永不过期 - 'prefix' => '', //缓存前缀,不宜过长 - 'persistent' => false, //是否长连接 false=短连接,推荐长连接 - ], - - 单例获取: - $redis = \think\Cache::connect(Config::get('cache')); - $redis->master(true); - $redis->get('key'); +配置参数: +'cache' => [ +'type' => 'Redisd' +'host' => 'A:6379,B:6379', //redis服务器ip,多台用逗号隔开;读写分离开启时,默认写A,当A主挂时,再尝试写B +'slave' => 'B:6379,C:6379', //redis服务器ip,多台用逗号隔开;读写分离开启时,所有IP随机读,其中一台挂时,尝试读其它节点,可以配置权重 +'port' => 6379, //默认的端口号 +'password' => '', //AUTH认证密码,当redis服务直接暴露在外网时推荐 +'timeout' => 10, //连接超时时间 +'expire' => false, //默认过期时间,默认0为永不过期 +'prefix' => '', //缓存前缀,不宜过长 +'persistent' => false, //是否长连接 false=短连接,推荐长连接 +], + +单例获取: +$redis = \think\Cache::connect(Config::get('cache')); +$redis->master(true); +$redis->get('key'); */ /** * ThinkPHP Redis简单主从实现的高可用方案 - * + * * 扩展依赖:https://github.com/phpredis/phpredis - * + * * 一主一从的实践经验 * 1, A、B为主从,正常情况下,A写,B读,通过异步同步到B(或者双写,性能有损失) * 2, B挂,则读写均落到A * 3, A挂,则尝试升级B为主,并断开主从尝试写入(要求开启slave-read-only no) * 4, 手工恢复A,并加入B的从 - * + * * 优化建议 * 1,key不宜过长,value过大时请自行压缩 * 2,gzcompress在php7下有兼容问题 - * + * * @todo * 1, 增加对redisCluster的兼容 * 2, 增加tp5下的单元测试 - * + * * @author 尘缘 <130775@qq.com> */ class Redisd { protected static $redis_rw_handler; protected static $redis_err_pool; - + protected $options = [ 'host' => '127.0.0.1', 'slave' => '', @@ -72,10 +72,10 @@ class Redisd 'length' => 0, 'prefix' => '', ]; - + /** * 为了在单次php请求中复用redis连接,第一次获取的options会被缓存,第二次使用不同的$options,将会无效 - * + * * @param array $options 缓存参数 * @access public */ @@ -85,25 +85,25 @@ class Redisd throw new Exception('_NOT_SUPPERT_:redis'); } - $this->options = $options = array_merge($this->options, $options); - $this->options ['func'] = $options ['persistent'] ? 'pconnect' : 'connect'; + $this->options = $options = array_merge($this->options, $options); + $this->options['func'] = $options['persistent'] ? 'pconnect' : 'connect'; - $host = explode(",", trim($this->options ['host'], ",")); - $host = array_map("trim", $host); - $slave = explode(",", trim($this->options ['slave'], ",")); + $host = explode(",", trim($this->options['host'], ",")); + $host = array_map("trim", $host); + $slave = explode(",", trim($this->options['slave'], ",")); $slave = array_map("trim", $slave); - $this->options ["server_slave"] = empty($slave) ? $host : $slave; - $this->options ["servers"] = count($slave); - $this->options ["server_master"] = array_shift($host); - $this->options ["server_master_failover"] = $host; + $this->options["server_slave"] = empty($slave) ? $host : $slave; + $this->options["servers"] = count($slave); + $this->options["server_master"] = array_shift($host); + $this->options["server_master_failover"] = $host; } - + /** * 主从选择器,配置多个Host则自动启用读写分离,默认主写,随机从读 * 随机从读的场景适合读频繁,且php与redis从位于单机的架构,这样可以减少网络IO * 一致Hash适合超高可用,跨网络读取,且从节点较多的情况,本业务不考虑该需求 - * + * * @access public * @param bool $master true 默认主写 */ @@ -115,26 +115,26 @@ class Redisd //如果不为主,则从配置的host剔除主,并随机读从,失败以后再随机选择从 //另外一种方案是根据key的一致性hash选择不同的node,但读写频繁的业务中可能打开大量的文件句柄 - if(!$master && $this->options["servers"] > 1) { + if (!$master && $this->options["servers"] > 1) { shuffle($this->options["server_slave"]); $host = array_shift($this->options["server_slave"]); - }else{ + } else { $host = $this->options["server_master"]; } $this->handler = new \Redis(); - $func = $this->options ['func']; - + $func = $this->options['func']; + $parse = parse_url($host); $host = isset($parse['host']) ? $parse['host'] : $host; - $port = isset($parse['host']) ? $parse['port'] : $this->options ['port']; - - $this->handler->$func($host, $port, $this->options ['timeout']); + $port = isset($parse['host']) ? $parse['port'] : $this->options['port']; - if ($this->options ['password'] != null) { - $this->handler->auth($this->options ['password']); + $this->handler->$func($host, $port, $this->options['timeout']); + + if (null != $this->options['password']) { + $this->handler->auth($this->options['password']); } - + APP_DEBUG && Log::record("[ CACHE ] INIT Redisd : {$host}:{$port} master->" . var_export($master, true), 'info'); //发生错误则摘掉当前节点 @@ -143,21 +143,21 @@ class Redisd } catch (\RedisException $e) { //phpredis throws a RedisException object if it can't reach the Redis server. //That can happen in case of connectivity issues, if the Redis service is down, or if the redis host is overloaded. - //In any other problematic case that does not involve an unreachable server + //In any other problematic case that does not involve an unreachable server //(such as a key not existing, an invalid command, etc), phpredis will return FALSE. - + Log::record(sprintf("redisd->%s:%s:%s", $master ? "master" : "salve", $host, $e->getMessage()), Log::ALERT); - + //主节点挂了以后,尝试连接主备,断开主备的主从连接进行升主 - if($master) { - if(! count($this->options["server_master_failover"])) { - throw new Exception("redisd master: no more server_master_failover. {$host} : ".$e->getMessage()); + if ($master) { + if (!count($this->options["server_master_failover"])) { + throw new Exception("redisd master: no more server_master_failover. {$host} : " . $e->getMessage()); return false; } $this->options["server_master"] = array_shift($this->options["server_master_failover"]); $this->master(); - + Log::record(sprintf("master is down, try server_master_failover : %s", $this->options["server_master"]), Log::ERROR); //如果是slave,断开主从升主,需要手工同步新主的数据到旧主上 @@ -165,23 +165,24 @@ class Redisd //$this->handler->slaveof(); } else { //尝试failover,如果有其它节点则进行其它节点的尝试 - foreach ($this->options["server_slave"] as $k=>$v) - { - if (trim($v) == trim($host)) + foreach ($this->options["server_slave"] as $k => $v) { + if (trim($v) == trim($host)) { unset($this->options["server_slave"][$k]); + } + } - + //如果无可用节点,则抛出异常 - if(! count($this->options["server_slave"])) { + if (!count($this->options["server_slave"])) { Log::record("已无可用Redis读节点", Log::ERROR); - throw new Exception("redisd slave: no more server_slave. {$host} : ".$e->getMessage()); + throw new Exception("redisd slave: no more server_slave. {$host} : " . $e->getMessage()); return false; } else { Log::record("salve {$host} is down, try another one.", Log::ALERT); return $this->master(false); } } - } catch(\Exception $e) { + } catch (\Exception $e) { throw new Exception($e->getMessage(), $e->getCode()); } @@ -190,19 +191,17 @@ class Redisd /** * 读取缓存 - * + * * @access public * @param string $name 缓存key * @return mixed */ public function get($name) { - Cache::$readTimes++; - $this->master(false); try { - $value = $this->handler->get($this->options ['prefix'] . $name); + $value = $this->handler->get($this->options['prefix'] . $name); } catch (\RedisException $e) { unset(self::$redis_rw_handler[0]); $this->master(); @@ -217,12 +216,12 @@ class Redisd $jsonData = $value ? json_decode($value, true) : $value; } - return ($jsonData === NULL) ? $value : $jsonData; + return (null === $jsonData) ? $value : $jsonData; } - + /** * 写入缓存 - * + * * @access public * @param string $name 缓存key * @param mixed $value 缓存value @@ -231,26 +230,24 @@ class Redisd */ public function set($name, $value, $expire = null) { - Cache::$writeTimes++; - $this->master(true); - - if (is_null($expire )) { - $expire = $this->options ['expire']; + + if (is_null($expire)) { + $expire = $this->options['expire']; } - $name = $this->options ['prefix'] . $name; - + $name = $this->options['prefix'] . $name; + /** * 兼容历史版本 * Redis不支持存储对象,存入对象会转换成字符串 * 但在这里,对所有数据做json_decode会有性能开销 */ - $value = (is_object($value) || is_array($value )) ? json_encode($value) : $value; - - if ($value === null) { - return $this->handler->delete($this->options ['prefix'] . $name); + $value = (is_object($value) || is_array($value)) ? json_encode($value) : $value; + + if (null === $value) { + return $this->handler->delete($this->options['prefix'] . $name); } - + // $expire < 0 则等于ttl操作,列为todo吧 try { if (is_int($expire) && $expire) { @@ -268,62 +265,61 @@ class Redisd return $result; } - + /** * 返回句柄对象 * 需要先执行 $redis->master() 连接到 DB - * + * * @access public * @return object */ - function handler() + public function handler() { return $this->handler; } - + /** * 删除缓存 - * + * * @access public * @param string $name 缓存变量名 * @return boolen */ public function rm($name) { - Cache::$writeTimes++; $this->master(true); - return $this->handler->delete($this->options ['prefix'] . $name); + return $this->handler->delete($this->options['prefix'] . $name); } - + /** * 清除缓存 - * + * * @access public * @return boolen */ public function clear() { - Cache::$writeTimes++; - $this->master(true); - return $this->handler->flushDB (); + return $this->handler->flushDB(); } - + /** * 析构释放连接 - * + * * @access public */ public function __destruct() { //该方法仅在connect连接时有效 - //当使用pconnect时,连接会被重用,连接的生命周期是fpm进程的生命周期,而非一次php的执行。 + //当使用pconnect时,连接会被重用,连接的生命周期是fpm进程的生命周期,而非一次php的执行。 //如果代码中使用pconnect, close的作用仅是使当前php不能再进行redis请求,但无法真正关闭redis长连接,连接在后续请求中仍然会被重用,直至fpm进程生命周期结束。 - + try { - if(method_exists($this->handler, "close")) - $this->handler->close (); + if (method_exists($this->handler, "close")) { + $this->handler->close(); + } + } catch (\Exception $e) { } } -} \ No newline at end of file +}