最近在做業務的時候需要實現客戶下單之後訂單逾時未支付自動取消的功能,剛開始確認了幾種方法:用戶端到時間請求取消服務端定時查詢有沒有需要取消的訂單,然後批量處理,下單後建立定時器,延時處理使用
redis 或者
memcache 儲存,設定到期時間,自動刪除。
綜合考慮上述方法,第一種最先排除,因為如果客戶把APP後台禁止或者網路連接禁止,那麼就無法發給服務端請求,訂單就會一直是未處理狀態;第二種方法使用的比較多,不過存在準確度的問題,還有需要確認定時任務的周期,暫時列為後補方法;第四種方法存在的問題就是訂單如果刪除就是物理刪除,無法統計未處理資料(當然可以存redis時候順便存在mysql這樣的資料庫做長久儲存然後用方法二定時處理)。
最終準備使用方法三。
再確認使用方法3的時候,由於使用的PHP這種開發語言,所以想實現定時器功能需要藉助 Swoole 或者 workerman 。由於 Swoole 是 C 開發的擴充架構,效能方面肯定比較好,就選了 Swoole 。
前期準備
使用Swoole首先需要在伺服器上安裝 Swoole 擴充,安裝方法和安裝其他擴充大同小異,可以參考這邊文章
安裝完之後檢測下擴充是否正常安裝,查看 phpinfo 或者 PHP-m ,如果出現 Swoole ,則說明安裝成功
Swoole 官方文檔有定時器的 相關文檔
開始測試
我們建立一個 swoole_test.php 檔案和一個 log.txt 檔案(用來測試), swoole_test.php代碼如下:
<?phpswoole_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檔案,結果如下:
然後在Linux終端運行PHP: /usr/local/php7/bin/php /home/app/swoole_test.php ,結果如下:
內心一陣。。。
原來定時器只能在 cli 模式下,那麼這個想法怕是要GG了,難道就栽倒這裡了嗎,難道就沒有別的方法了嗎?就在我欲哭無淚的時候突然靈光乍現,一個詞閃到我的腦海: Python !
對,我們不能單單靠著 PHP 啊,還有 Python 這種神奇的語言呢,我們知道 Python 的 os 模組裡的 os.system 方法是可以執行命令列的,那麼不就可以實現在 cli 模式下運行剛才的 swoole_test.php 檔案了麼。
內心一陣激動後,覺得測試是否可行
我們知道 Linux 都是內建 Python 的,但是不同的版本 Python 版本不同,有的內建的是 Python2.6 ,版本過低了,所以需要裝一個高版本的,這裡我選擇 Python3 ,注意不要覆蓋系統內建的 Python2 。以下是大致的安裝步驟:
接下來終端輸入: Python3 ,如果出現
則安裝成功。
安裝完 Python3 之後,我們建立一個 test.py 檔案,內容如下:
#!usr/bin/env python3`#-*- coding:utf-8 -*-import osret = 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 是可以實現 PHP 的 cli 模式的。┗|`O′|┛ 嗷~~
到這裡就會有同學疑惑了,你這使用 Python 實現了 PHP 的 cli 模式,但是怎麼通過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 檔案內容:
<?phpecho "begin<br>";$program="/usr/local/php7/bin/php /home/app/nongyephp/swoole_test.php"; #注意使用絕對路徑(exec ($program));echo "end<br>";die;
在直接存取 test.php 檔案,反饋結果和藉助 Python 一樣,這樣就可以免去 Python 那一步,直接用 PHP 的 exec 函數來執行 PHP 檔案。
結尾
測試通過後發現這種方法是可以建立定時器並且通過web遠程使用的,不過有個問題,如果用和我上述一樣用網頁類比會發現網頁重新整理是要等 test.php 執行完才會結束,也就是說如果我們把延時器的時間設成30分鐘會要等待30分鐘才會有反饋資訊,這種方式肯定行不通的,所以需要使用非同步訪問,比如使用web的 ajax 技術和其他非同步技術,這裡不再贅述
尾巴
以上只是我想到解決問題的想法和實施步驟,到了真正開發可能不會選擇這種方式,因為沒有經過效能測試,而且對於進程式控制制和線程式控制制並沒有多深入的瞭解,所以以後做訂單自動取消還是會選擇方法2的吧。
上述方法其實完全可以省掉 Python 那一步,我沒有去掉的原因是把我的實現經曆寫出來,因為我覺得開發期間可能真的會遇到這種多此一舉的方式,總之是要多思考,多看代碼,找出能最佳化的方案,這裡感覺自己差得很遠,共勉吧