PHP電商訂單自動確認收貨redis隊列

来源:http://www.cnblogs.com/doyourself/archive/2017/05/16/6861374.html
-Advertisement-
Play Games

一、場景 之前做的電商平臺,用戶在收到貨之後,大部分都不會主動的點擊確認收貨,導致給商家結款的時候,商家各種投訴,於是就根據需求,要做一個訂單在發貨之後的x天自動確認收貨。所謂的訂單自動確認收貨,就是在在特定的時間,執行一條update語句,改變訂單的狀態。 二、思路 最笨重的做法,通過linux後 ...


一、場景

之前做的電商平臺,用戶在收到貨之後,大部分都不會主動的點擊確認收貨,導致給商家結款的時候,商家各種投訴,於是就根據需求,要做一個訂單在發貨之後的x天自動確認收貨。所謂的訂單自動確認收貨,就是在在特定的時間,執行一條update語句,改變訂單的狀態。

二、思路

最笨重的做法,通過linux後臺定時任務,查詢符合條件的訂單,然後update。最理想情況下,如果每分鐘都有需要update的訂單,這種方式也還行。奈何平臺太小,以及賣家發貨時間大部分也是密集的,不會分散在24小時的每分鐘。那麼,定時任務的話,查詢過多,不適合。這裡可以先把將要自動確認收貨的訂單信息存儲到其他介質上,比如redis,memcache,rabbitmq,然後執行的腳本從前面的介質獲取到訂單信息來判斷,這裡可以大大的減少資料庫的查詢壓力。

redis隊列的生產者

對此,我們選擇每天在凌晨兩點的時候,通過linux的定時任務把即將要確認收貨的訂單信息查詢出來,然後存儲在redis上,redis上我們選擇的隊列,隊列處理的特點就是先進先出,前面的數據在查詢訂單時,通過發貨時間排序,所以最先出隊列的肯定是距離規定的自動收貨時間最近的訂單。代碼如下

$successCount=0;
$failCount=0;
$screen_time = 3600*24*9;//設置篩選天數
$data = array();
$now_time = time();
//查詢符合要求的數據
$sql="select id,send_time as deliver_time from `order` where is_send=1 and is_del=0 and is_cancel=0 and is_token=0 and send_time>0 and send_time + {$screen_time} < $now_time 
order by send_time asc";
$res = $con->query($sql);
//當隊列還有數據時將數據記錄並清除
while($redis->LLEN('auto_recevice_order')){
$txt = '執行時間:'.date('Y-m-d H:i:s').',信息:'.$redis->RPOP('auto_recevice_order');
file_put_contents('./autoToken/fail_log.txt',$txt."\r\n".PHP_EOL,FILE_APPEND);
$failCount++;
}
//重新填充數據進隊列
while ($row = $res->fetch_assoc()) {
 $successCount++;
 $redis->LPUSH('auto_recevice_order',json_encode($row1));
}
 $con->close(); 
 $success=date('Y-m-d H:i:s').':[推送成功]:本次成功推送數據:'.$successCount.'條;記錄上次處理失敗數據:'.$failCount."條\r\n";
 file_put_contents('./success_log.txt',$success."\r\n".PHP_EOL,FILE_APPEND);

  

redis隊列的消費者

隊列的消費者沒有通過linux的定時任務去做,用linux的screen+php cli模式執行php腳本,消費者只需要不斷的從隊列中讀取訂單信息,然後判斷訂單信息中的發貨時間,如果達到自動收貨的要求,就執行update語句。同時如果沒有達到收貨的時間,而且與收貨時間間距比較大的時候,可以讓php腳本休眠sleep一定的時間數,這個時間數自己調節設計,獲取出來的未達到時間要求的訂單,需要重新推送到redis隊列中去,而且還是隊列的頂端。以便下次獲取。代碼如下:

$set_time = 3600*24*10;//設置幾天後自動收貨
while(true){
if($i%30==0){
usleep(10);//防止while 迴圈使CPU使用率過高
}
if($redis->LLEN('auto_recevice_order')){
$data = json_decode($redis->RPOP('auto_recevice_order'));
$id = (int)$data->id;//將數據轉化為整形
$deliver_time = (int)$data->deliver_time;//將數據轉化為整形
$res1 = $res2 =false;
$now_time = time();
if(($deliver_time+$set_time)<$now_time){
  $sql1 = "update `order` set `is_token`='1',`token_time` = $now_time where id=$id and is_send=1 and is_del=0 and is_cancel=0 and is_token=0 and send_time + {$set_time} < $now_time";
  $res1 = $con->query($sql1);//更新數據
$rows = mysqli_affected_rows($con);
if($rows){
 $ip = $this->getIp();
 $sql2 = "insert into `order_log`(`order_id`,`log_msg`,`log_ip`,`log_role`,`log_user`,`log_order_state`,`log_time`) VALUES($id,'系統自動收貨','$ip','系統','伺服器','收貨',$now_time)";//寫入訂單日誌
 $res2 = $con->query($sql2);//添加日誌數據
   }
 }
 if($res1==false){//將沒達到條件的數據重新插入隊列中
     $redis->RPUSH('auto_recevice_order',json_encode(array('id'=>$id,'deliver_time'=>$deliver_time)));
   }
}
 $i++;
}

這裡執行php腳本,需要用到linux的screen或者supervisor、nohup守護進程。具體用法可自行百度.同樣腳本裡面最好有必須的日誌記錄。

三、思考

隨著業務的增長,在隊列中同一秒內,存在的多個需要處理的訂單,而一次只能從隊列中取出一個相關訂單信息的時候,可以採用一個生產者多個消費者的模式,這種情況下,可以用到鎖機制,保證一條消息只能到達一個消費者。當redis數據達到一定的量之後,也可以適當的調整生產者的執行頻率和對應的條件。

更多原創博客請看個人獨立博客:傳送門


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、Map用法 ...
  • 數組排序演算法:冒泡排序法、選擇排序法、直接插入排序法、快速排序法 ...
  • 雲服務 子系統:後臺管理系統、Restfu服務系統、Dubbo服務/管控/監控中心 Zookeeper註冊中心、報表分析系統、日誌記錄系統、定時調度系統 搜索引擎系統、分散式文件系統、消息系統、SSO單點登錄系統 SOA管控平臺、UI組件系統、OA辦公系統、CMS新聞發佈系統 支付系統、資料庫配置系 ...
  • dropzone.js預設是Ajax上傳圖片給伺服器,那麼如何獲取到圖片名呢?其實我們是可以通過dropzone的success函數獲取到伺服器返回的數據 dropzone.js在HTML的配置如下; ...
  • ———————————————————————————————————————————— 實驗7.4:RS232串口應用實例 設計要求: 通過串口發送接收數據(串口工作方式1) 實現思路: 定時器設置 串口設置 工作方式 波特率 開啟定時/計數器 串口執行 接收數據 發送數據 串口設置 工作方式 波 ...
  • 路由配置 路由組 允許共用路由屬性,例如中間件和命名空間等,我們沒有必要為每個路由單獨設置共有屬性,共有屬性會以數組的形式放到 Route::group 方法的第一個參數中。 中間件('middleware' => 'auth'),驗證用戶是否登錄。 命名空間 ('namespace' => 'Ad ...
  • spring mvc Spring框架(框架即:編程註解+xml配置的方式)MVC是Spring框架的一大特征,Spring框架有三大特征(IOC(依賴註入),AOP(面向切麵),MVC(建模M-視圖V-控制器C)。框架一般用於團隊開發,使用分層的方式使每個人完成不同的模塊,然後再組合在一起,使完成 ...
  • 變數 在程式設計中,變數(英語:Variable,scalar)是指一個包含部分已知或未知數值或資訊(即一個值)之儲存位址,以及相對應之符號名稱(識別字)。通常使用變數名稱參照儲存值;將名稱和內容分開能讓被使用的名稱獨立於所表示的精確訊息之外。電腦原始碼中的識別字能在執行期間綁扎一個值,且該變數的值 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...