PHP實現訂單的延時處理

真沒想到本身的這篇很水的文章還有一些可愛的朋友們去閱讀,真的很感謝大家。其實這篇文章是我在元旦假期最後一天無事可作的時候偶然腦子抽搐想到的一個場景,而後試着去實現的,其實訂單自動取消功能網上有不少可借鑑的方法,包括評論區的朋友們提到的定時任務和消息隊列,我這個只是在一個本身想到的方法(我認可真的真的很水)。
感謝評論區的朋友們的熱心建議和教導,我也會去借鑑一下大家的指導意見,同時繼續學習,再次對大家表示感謝~ღ( ´・ᴗ・` )比心php


如下是原文:html

業務需求

最近在作業務的時候須要實現客戶下單以後訂單超時未支付自動取消的功能,剛開始確認了幾種方法:python

  • 客戶端到時間請求取消
  • 服務端定時查詢有沒有須要取消的訂單,而後批量處理
  • 下單後建立定時器,延時處理
  • 使用redis或者memcache存儲,設置過時時間,自動刪除

綜合考慮上述方法,第一種最早排除,由於若是客戶把APP後臺禁止或者網絡鏈接禁止,那麼就沒法發給服務端請求,訂單就會一直是未處理狀態;第二種方法使用的比較多,不過存在準確度的問題,還有須要確認定時任務的週期,暫時列爲後補方法;第四種方法存在的問題就是訂單若是刪除就是物理刪除,沒法統計未處理數據(固然能夠存redis時候順便存在mysql這樣的數據庫作長久存儲而後用方法二定時處理)。
最終準備使用方法三。
再確認使用方法3的時候,因爲使用的PHP這種開發語言,因此想實現定時器功能須要藉助Swoole或者workerman。因爲SwooleC開發的擴展框架,性能方面確定比較好,就選了Swoolemysql

前期準備

  • 使用Swoole首先須要在服務器上安裝Swoole擴展,安裝方法和安裝其餘擴展大同小異,能夠參考這邊文章
  • 安裝完以後檢測下擴展是否正常安裝,查看phpinfo或者PHP-m,若是出現Swoole,則說明安裝成功
  • Swoole官方文檔有定時器的相關文檔

開始測試

咱們建立一個swoole_test.php文件和一個log.txt文件(用來測試),swoole_test.php代碼以下:web

<?php
swoole_timer_after(3000, function () {
    append_log(time());
    echo "after 3000ms.\n";
});

function append_log($str) {
    $dir = 'log.txt';
    $fh = fopen($dir, "a");
    fwrite($fh, $str."\n");
    fclose($fh);
}

而後在網頁訪問這個PHP文件,結果以下:圖片描述ajax

而後在Linux終端運行PHP:/usr/local/php7/bin/php /home/app/swoole_test.php,結果以下:
圖片描述
心裏一陣。。。
圖片描述
原來定時器只能在cli模式下,那麼這個想法怕是要GG了,難道就栽倒這裏了嗎,難道就沒有別的方法了嗎?就在我欲哭無淚的時候忽然靈光乍現,一個詞閃到個人腦海:Python
對,咱們不能單單靠着PHP啊,還有Python這種神奇的語言呢,咱們知道Pythonos模塊裏的os.system方法是能夠執行命令行的,那麼不就能夠實如今cli模式下運行剛纔的swoole_test.php文件了麼。
心裏一陣激動後,以爲測試是否可行
咱們知道Linux都是自帶Python的,可是不一樣的版本Python版本不一樣,有的自帶的是Python2.6,版本太低了,因此須要裝一個高版本的,這裏我選擇Python3,注意不要覆蓋系統自帶的Python2 。如下是大體的安裝步驟:redis

  • wget http://python.org/ftp/python/...
  • tar xf Python-3.6.0.tar.xz
  • cd Python-3.6.0
  • ./configure --prefix=/usr/local/python3
  • make && make install
  • ln -s /usr/local/python3/bin/python3 /usr/bin/python3

接下來終端輸入:Python3,若是出現
圖片描述sql

則安裝成功。
安裝完Python3以後,咱們新建一個test.py文件,內容以下:數據庫

#!usr/bin/env python3`
#-*- coding:utf-8 -*-

import os
ret = os.system("/usr/local/php7/bin/php /home/app/swoole_test.php") #請使用本身系統的絕對路徑
print(ret)

而後咱們在終端執行:/usr/bin/python3 /home/app/test.py,注意:這裏只是執行PHP文件,可是文件裏的echo內容是不會在終端輸出的,這時候就用到剛纔新建的log.txt文件了。執行完Python文件後,咱們去log文件檢查下,發現內容已經寫入,因此使用Python是能夠實現PHPcli模式的。┗|`O′|┛ 嗷~~
圖片描述segmentfault

到這裏就會有同窗疑惑了,你這使用Python實現了PHPcli模式,可是怎麼經過web遠程訪問呢?這個時候就用到PHP的exec方法了,咱們知道PHP的exec方法和Python的os.system方法同樣是能夠執行命令行命令的,因此咱們能夠新建一個test.php文件,內容以下:

<?php
$program="/usr/bin/python3 /home/app/nongyephp/test.py"; #注意使用絕對路徑
echo "begin<br>";
(exec ($program));
echo "end<br>";
die;

而後咱們經過網頁訪問test.php文件。結果以下:
圖片描述

而後去log文件檢查,發現也寫入日誌了,因此這個方法是可行的!
作到這裏內心美滋滋的,不過老以爲好像哪裏不對,終於終於意識到一個很傻逼的問題:既然PHP能夠直接有命令行函數,爲啥畫蛇添足藉助Python而後在用Python的函數呢?這不是脫了褲子放屁畫蛇添足嗎?
再大罵本身是傻逼N遍以後,我默默修改了test.php文件內容:

<?php
echo "begin<br>";
$program="/usr/local/php7/bin/php /home/app/nongyephp/swoole_test.php"; #注意使用絕對路徑
(exec ($program));
echo "end<br>";
die;

在直接訪問test.php文件,反饋結果和藉助Python同樣,這樣就能夠免去Python那一步,直接用PHPexec函數來執行PHP文件。

結尾

測試經過後發現這種方法是能夠建立定時器而且經過web遠程使用的,不過有個問題,若是用和我上述同樣用網頁模擬會發現網頁刷新是要等test.php執行完纔會結束,也就是說若是咱們把延時器的時間設成30分鐘會要等待30分鐘纔會有反饋信息,這種方式確定行不通的,因此須要使用異步訪問,好比使用web的ajax技術和其餘異步技術,這裏再也不贅述

尾巴

  • 以上只是我想到解決問題的想法和實施步驟,到了真正開發可能不會選擇這種方式,由於沒有通過性能測試,並且對於進程控制和線程控制並無多深刻的瞭解,因此之後作訂單自動取消仍是會選擇方法2的吧。
  • 上述方法其實徹底能夠省掉Python那一步,我沒有去掉的緣由是把個人實現經歷寫出來,由於我以爲開發期間可能真的會遇到這種畫蛇添足的方式,總之是要多思考,多看代碼,找出能優化的方案,這裏感受本身差得很遠,共勉吧
相關文章
相關標籤/搜索