av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁技術文章
文章詳情頁

詳解PHP解決守護進程Redis假死

瀏覽:27日期:2022-09-07 08:15:24
目錄一、一個簡單的守護進程示例二、一個不再假死(偽活)的 Redis 常駐進程示例一、一個簡單的守護進程示例

<?php$redis = new Redis();$redis->connect(’localhost’, 6379);$redis->auth(’xxxxx’); // Redis 密碼如果沒有設置為空字符串。$redis->select(1);$queueKey = ’redis_queue_services_key’; // 業務數據隊列。$queueIngKey = ’redis_queue_services_ing_key’; // 處理中的隊列。try { while (true) {$element = $redis->bRPopLPush($queueKey, $queueIngKey, 60);if ($element) { $data = json_decode($element, true); /** * ...... 此處省略業務邏輯 ...... * */} else { usleep(100000); // 睡眠 0.1 秒。} }} catch (Exception $e) { exit('Error:{$e->getMessage()}');}

這段代碼我們很容易看懂。

它就是通過Redis的阻塞方法bRPopLPush循環從 Redis 隊列中取出數據并處理。如果沒有取到數據就休眠一秒。之所以休眠是為了保證 CPU 能得到充分的利用。因為,我們已經使用了阻塞方法阻塞 60 秒。所以,這個位置休眠與否并不重要。

當我們的業務出現任何錯誤,我們通過try catch進行異常捕獲然后將錯誤信息直接輸出并退當前腳本。

博主寒冰第一次編寫常駐后臺運行的守護進程時,就是如上這種方式寫的代碼。結果,這段代碼運行到 30s 的時候報錯了。提示我們 socket 流超時。于是我在這個腳本頭部加了如下代碼:

ini_set(’default_socket_timeout’, -1);

這樣我們的PHP就不會主動段掉我們與 Redis 的 socket 連接了。

但是,好景不長。過了一段時間,大概半個月吧。運維同學告訴我 Redis 隊列的數據出現了未消費的情況。然后,我查看了消費日志。的確沒有產生新的消費日志。因為我有一個習慣,每個消費消費的時候都會把成功消費的日志寫到文件中。消費失敗的也寫入日志文件中。這樣,我就知道失敗的具體原因。

但是,這次我真的沒有發現有任何的錯誤發生。

常駐后臺進程處理存活狀態。并沒有變成孤兒進程。 常駐后臺進程內存也沒有出現泄漏。 系統 CPU/內存 資源都處理正在狀態。 系統打開的句柄資源也是低消狀態。 帶寬也處理低消狀態。 其它常駐進程也處理正常消費的工作狀態。也就排除了 Redis 故障的問題。

我當時也懷疑過是不是像MySQL一樣常時間連接不進行任何操作,服務器端會主動斷開連接。但是,MySQL 服務器端主動段掉連接會提示:MySQL server has gone away的錯誤。但是,我們的 Redis 服務器端沒有給我們報任何錯誤信息呀。

我們公司用的是阿里云的 Redis 產品。我懷疑是不是 Redis 版本太低造成的這個隱性 BUG。于是,我們將阿里云的 Redis 服務升級到了阿里云支持的最新版本。

結果還是失敗了。我們的 Redis 還是假死了?;蛘哒f我們的 Redis 處于偽活狀態。

你認為 Redis 活著,其實它早已經死了。你認為 Redis 死了,但是它卻沒有死亡的特征。

最后,我冷靜下來。

我假定此時的 Redis 已經死了。只是沒有告訴客戶端而已。那么我只需要每次檢測一下 Redis 連接是否存活就好了。

于是,我翻看了 Redis 的 API。發現它提供了一個ping()的方法來檢測連接是否存活。

二、一個不再假死(偽活)的 Redis 常駐進程示例

<?php$redis = new Redis();$redis->connect(’localhost’, 6379);$redis->auth(’xxxxx’); // Redis 密碼如果沒有設置為空字符串。$redis->select(1);$queueKey = ’redis_queue_services_key’; // 業務數據隊列。$queueIngKey = ’redis_queue_services_ing_key’; // 處理中的隊列。try { while (true) {$element = $redis->bRPopLPush($queueKey, $queueIngKey, 60);if ($element) { $data = json_decode($element, true); /** * ...... 此處省略業務邏輯 ...... * */} else { $pong = $redis->ping(); if ($pong != ’+PONG’) {throw new Exception(’Redis ping failure!’, 500); } usleep(100000); // 睡眠 0.1 秒。} }} catch (Exception $e) { exit('Error:{$e->getMessage()}');}

通過代碼對比,我們在第一版代碼的基礎上加了如下代碼:

$pong = $redis->ping();if ($pong != ’+PONG’) { throw new Exception(’Redis ping failure!’, 500);}

我們向 Redis 服務器發送ping的時候,服務器會返回+PONG字符串。當然,這個是 Redis 擴展封裝過的方法。真正的 ping 是不會有 + 號的。

當我們每次 ping 的時候,Redis 服務器就會認為我們的 Redis 客戶端連接處于存活狀態。就不會斷掉我們的連接了。

把代碼進行改造之后,假死頭痛的問題再也沒出現了。

以上就是詳解PHP解決守護進程Redis假死的詳細內容,更多關于PHP解決守護進程Redis假死的資料請關注好吧啦網其它相關文章!

標簽: PHP
相關文章:
主站蜘蛛池模板: 国产精品一区二区三区在线 | 国产69精品久久99不卡免费版 | 美女视频h | 国产精品视频网站 | 精品一二区| 日韩一区二区黄色片 | 精品一区二区三区91 | 国产精品视频一 | 九九九视频| 6080yy精品一区二区三区 | 日韩精品一区二区三区第95 | 一级特黄色毛片 | 日本人和亚洲人zjzjhd | 欧美日韩久久久久 | 久久久亚洲成人 | 欧美日韩一区二区三区四区 | 欧美一区二区另类 | 国产欧美在线播放 | 欧美成人免费在线视频 | 国产乱精品一区二区三区 | 国产精品久久久久久久7777 | 久久久久久色 | 久久国产精品免费 | 亚洲欧美精品 | 午夜欧美一区二区三区在线播放 | 久久国产精品视频观看 | 99爱国产| 天堂影院av | 国产伦精品一区二区三区视频金莲 | 成人亚洲性情网站www在线观看 | 婷婷中文字幕 | 成人在线视频观看 | 成人久久18免费网站麻豆 | 久热久热 | 日本亚洲精品成人欧美一区 | 精品欧美乱码久久久久久1区2区 | 亚洲国产精品福利 | 色视频www在线播放国产人成 | 久久精品免费一区二区三 | 国产草草视频 | 中文字幕一区二区三区四区五区 |