色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

PHP簡(jiǎn)易延時(shí)隊(duì)列的實(shí)現(xiàn)流程詳解

瀏覽:134日期:2022-06-05 17:00:41
目錄
  • 需求說(shuō)明
  • 設(shè)計(jì)思路
  • 實(shí)現(xiàn)
  • 最后說(shuō)點(diǎn)

需求說(shuō)明

  • 當(dāng)用戶申請(qǐng)售后,商家未在n小時(shí)內(nèi)處理,系統(tǒng)自動(dòng)進(jìn)行退款。
  • 商家拒絕后,用戶可申請(qǐng)客服介入,客服x天內(nèi)超時(shí)未處理,系統(tǒng)自動(dòng)退款。
  • 用戶收到貨物,x天自動(dòng)確認(rèn)收貨
  • 等等需要延時(shí)操作的流程……

設(shè)計(jì)思路

  • 設(shè)計(jì)一張隊(duì)列表,記錄所有隊(duì)列的參數(shù),執(zhí)行狀態(tài),重試次數(shù)
  • 將創(chuàng)建隊(duì)列的id 存于redis 中,使用zset有序集合。按照時(shí)間戳進(jìn)行排序
  • 使用croontab定時(shí)任務(wù)每分鐘執(zhí)行一次

實(shí)現(xiàn)

新建隊(duì)列表

CREATE TABLE `delay_queue` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `params` varchar(512) DEFAULT NULL,
  `message` varchar(255) DEFAULT "" COMMENT "執(zhí)行結(jié)果",
  `ext_string` varchar(255) DEFAULT "" COMMENT "擴(kuò)展字符串,可用于快速檢索。取消該隊(duì)列",
  `retry_times` int(2) DEFAULT "0" COMMENT "重試次數(shù)",
  `status` int(2) NOT NULL DEFAULT "1" COMMENT "1 待執(zhí)行, 10 執(zhí)行成功, 20 執(zhí)行失敗,30取消執(zhí)行",
  `created` datetime DEFAULT NULL,
  `modified` datetime DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `ext_idx` (`ext_string`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

部分隊(duì)列的操作方法,新增隊(duì)列、取消隊(duì)列、隊(duì)列執(zhí)行成功、隊(duì)列執(zhí)行失敗、隊(duì)列重試【重試時(shí)間間隔抄的微信支付的異步通知時(shí)間】

class DelayQueueService
{
    // 重試時(shí)間,最大重試次數(shù) 15
    private static $retryTimes = [
15, 15, 30, 3 * 60, 10 * 60, 20 * 60, 30 * 60, 30 * 60, 30 * 60, 60 * 60,
3 * 60 * 60, 3 * 60 * 60, 3 * 60 * 60, 6 * 60 * 60, 6 * 60 * 60,
    ];
	/**
	 * @description 增加隊(duì)列至redis
	 * @param $queueId
	 * @param int $delay 需要延遲執(zhí)行的時(shí)間。單位秒
	 * @return void
	 */
	public function addDelayQueue($queueId, int $delay)
	{
	    $time = time() + $delay;
	    $redis = RedisService::getInstance();
	    $redis->zAdd("delay_queue_job", $time, $queueId);
	}
	// 取消redis 隊(duì)列
	public function cancelDelayQueue($ext)
	{
	    $row = $query->getRow(); // 使用ext_string 快速檢索到相應(yīng)的記錄
	    if ($row) {
	$redis = RedisService::getInstance();
	$redis->zRem("delay_queue_job", $row->id);
	$row->status = DelayQueueTable::STATUS_CANCEL;
	$table->save($row);
	    }
	}
	/**
	 * @description 執(zhí)行成功
	 * @return void
	 */
	public static function success($id, $message = null)
	{
	    $table->update([
	"status" => DelayQueueTable::STATUS_SUCCESS,
	"message" => $message ?? "",
	"modified" => date("Y-m-d H:i:s"),
	    ], [
	"id" => $id,
	    ]);
	}
	/**
	 * @description 執(zhí)行失敗
	 * @return void
	 */
	public static function failed($id, $message = null)
	{
	    $table->updateAll([
	"status" => DelayQueueTable::STATUS_FAILED,
	"message" => $message ?? "",
	"modified" => date("Y-m-d H:i:s"),
	    ], [
	"id" => $id,
	    ]);
	}
	/**
	 * @description 失敗隊(duì)列重試,最大重試15次
	 * @param $id
	 * @return void
	 */
	public static function retry($id)
	{
	    $info = self::getById($id);
	    if (!$info) {
	return;
	    }
	    $retryTimes = ++$info["retry_times"];
	    if ($retryTimes > 15) {
	return;
	    }
	    $entity = [
	"params" => $info["params"],
	"ext_string" => $info["ext_string"],
	"retry_times" => $retryTimes,
	    ];
	    $queueId = $table->save($entity);
	    self::addDelayQueue($queueId, self::$retryTimes[$retryTimes - 1]);
	}
}

在命令行進(jìn)行任務(wù)的運(yùn)行

public function execute(Arguments $args, ConsoleIo $io)
{
    $startTimestamp = strtotime("-1 days");
    $now = time();
    $redis = RedisService::getInstance();
    $queueIds = $redis->zRangeByScore("delay_queue_job", $startTimestamp, $now);
    if ($queueIds) {
foreach ($queueIds as $id) {
    $info = // 按照隊(duì)列id 獲取相應(yīng)的信息
    if ($info["status"] === DelayQueueTable::STATUS_PADDING) {
$params = unserialize($info["params"]); // 創(chuàng)建記錄的時(shí)候,需要試用serialize 將類名,方法,參數(shù)序列化
$class = $params["class"];
$method = $params["method"];
$data = $params["data"];
try {
    call_user_func_array([$class, $method], [$data]);
    $redis->zRem("delay_queue_job", $id);
    $msg = date("Y-m-d H:i:s") . " [info] success: $id";
    DelayQueueService::success($id, $msg);
    $io->success($msg);
} catch (Exception $e) {
    $msg = date("Y-m-d H:i:s") . " [error] {$e->getMessage()}";
DelayQueueService::failed($id, $msg);
// 自定義異常code,不進(jìn)行隊(duì)列重試
if (10000 != $e->getCode()) {
    DelayQueueService::retry($id);
}
$io->error($msg);
}
    }
}
    }
}

最后說(shuō)點(diǎn)

  • 我這邊的系統(tǒng)對(duì)實(shí)時(shí)性要求不高,所以直接使用的是linuxcrond 服務(wù),每分鐘運(yùn)行一次。如需精確到秒級(jí),可寫一個(gè)shell,一分鐘循環(huán)執(zhí)行<=60
  • 因?yàn)槟壳暗臄?shù)據(jù)較少,延時(shí)隊(duì)列加入的只有小部分。所以就在command 里面直接執(zhí)行更新操作了,后期如果隊(duì)列多,且有比較耗時(shí)的操作,可考慮把耗時(shí)操作單獨(dú)放置一個(gè)隊(duì)列中。本方法只用于將數(shù)據(jù)塞進(jìn)隊(duì)列。

附上 shell 腳本 一分鐘執(zhí)行60次

#!/bin/bash
step=2 #間隔的秒數(shù),不能大于60
for (( i = 0; i < 60; i=(i+step) )); do
   echo $i # do something
   sleep $step
done 

到此這篇關(guān)于PHP簡(jiǎn)易延時(shí)隊(duì)列的實(shí)現(xiàn)流程詳解的文章就介紹到這了,更多相關(guān)PHP延時(shí)隊(duì)列內(nèi)容請(qǐng)搜索以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持!

標(biāo)簽: PHP
主站蜘蛛池模板: 2022日韩理论片在线观看 | 久久精品视频在线观看榴莲视频 | 久久综合久久久 | 欧美日本视频一区 | 久久凹凸| a级国产乱理伦片在线观看国 | 天码毛片一区二区三区入口 | 久久厕所 | 日韩在线一区二区三区 | 欧美久久久久久 | 国产中文久久精品 | 亚洲一区二区三区精品视频 | 亚洲高清色 | 国产伦精品一区二区三区四区 | 黄色wwwxxx| 中文字幕在线免费观看 | 国产微拍精品福利视频 | 国产免费播放一区二区 | 一级一片免费播放 | 欧美国产日韩在线 | 欧美俄罗斯一级毛片激情 | 久久只有这才是精品99 | 欧美黑人xxxxxxxxxx | 欧美精品亚洲一区二区在线播放 | 精品久久九九 | 波多野结衣视频在线观看地址免费 | 日韩国产成人资源精品视频 | 中文字幕亚洲精品 | 日韩免费一级a毛片在线播放一级 | 亚洲精品日本 | 国产成人看片免费视频观看 | 日本一区二区三区高清在线观看 | 亚洲精品综合欧美一区二区三区 | 最新亚洲精品国自产在线 | 午夜免费片在线观看不卡 | 国产精彩视频在线 | 在线高清国产 | 一级特黄色毛片免费看 | 九九热视频精品在线观看 | 国产精品成人一区二区不卡 | 欧美亚洲一区二区三区四 |