Redis 異常 read error on connection 的解決方案

一、造成原因與場景

在接口連接 redis 時,使用瞭 pconnect 進行 redis 長連接,在 Redis->auth() 中拋出異常:

PHP Fatal error:  Uncaught exception 'RedisException' with message 'read error on connection' 

註:由於 php-fpm 對連接的重用性,所以,當存在cgi進程管理器[如php-fpm,spawn-cgi],使用 pconnect 效率比 connect 高。但是由於每個持久連接會占用內存,當並發量很大的時,很可能造成內存吃緊

二、出現原因

通過 pconnect 的 redis 長連接不會主動釋放,會一直保持在php-fpm進程中,等待完成下一個請求的任務,即使通過 close 也僅是使當前php不能再進行redis請求,但無法真正關閉redis長連接,連接在後續請求中仍然會被重用,直至fpm進程生命周期結束。

這樣會出現以下兩種情況:

1)原因1 使用瞭已經斷開的連接

假如這個連接已經斷開瞭,下個請求可能直接使用上個斷開的連接,當 php-fpm 處理下一個請求 reuse 一個斷開的連接可能導致 read error on connection 的異常,以下是 phpredis 源碼的解釋:

2)原因2:執行超時

  • a)客戶端設置的超時時間過短導致的;
  • b)客戶端未設置超時時間,但是服務端執行時間超過瞭默認超時時間設置

php 有一個配置項 default_socket_timeout 表示一個 socket 連接的超時時間,默認是60s,而phpredis 這個擴展的底層連接 redis 是使用的 socket,所以當一個連接在60秒內沒有被使用的時候,下次連接就會拋出這個異常。

三、解決方案

1)如果是原因1

長連接改為短連接 或者 在 pconnect 中增加超時時間

$this->redis->pconnect($host, $port, 2);       //@todo 2秒後該連接超時,將使用新的長連接

2)如果是原因2:修改超時時間

(不推薦) 方案1:

php.ini 中設置

default_socket_timeout = -1

然後重啟php-fpm

方案2:

動態修改 php.ini 

ini_set('default_socket_timeout', -1); // 不超時

方案3:

設置 redis 的超時時間為不超時:

$this->redis->setOption(Redis::OPT_READ_TIMEOUT, -1);

到此這篇關於Redis 異常 read error on connection 的解決方案的文章就介紹到這瞭,更多相關Redis 異常 內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: