業務需求
訂單是我們在日常開發中經常會遇到的一個功能,最近在做業務的時候需要實現客戶下單之后訂單超時未支付自動取消的功能,剛開始確認了幾種方法:
- 客戶端到時間請求取消
- 服務端定時查詢有沒有需要取消的訂單,然后批量處理
- 下單后創建定時器,延時處理
- 使用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代碼如下:
?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文件,結果如下:
然后在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 。以下是大致的安裝步驟:
- wget http://python.org/ftp/python/3.6.0/Python-3.6.0.tar.xz
- 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,如果出現

則安裝成功。
安裝完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是可以實現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 "beginbr>";
(exec ($program));
echo "endbr>";
die;
然后我們通過網頁訪問test.php文件。結果如下:

然后去log文件檢查,發現也寫入日志了,所以這個方法是可行的!
做到這里心里美滋滋的,不過老覺得好像哪里不對,終于終于意識到一個很傻逼的問題:既然PHP可以直接有命令行函數,為啥多此一舉借助Python然后在用Python的函數呢?這不是脫了褲子放屁多此一舉嗎?
再大罵自己是傻逼N遍之后,我默默修改了test.php文件內容:
?php
echo "beginbr>";
$program="/usr/local/php7/bin/php /home/app/nongyephp/swoole_test.php"; #注意使用絕對路徑
(exec ($program));
echo "endbr>";
die;
在直接訪問test.php文件,反饋結果和借助Python一樣,這樣就可以免去Python那一步,直接用PHP的exec函數來執行PHP文件。
結尾
測試通過后發現這種方法是可以創建定時器并且通過web遠程使用的,不過有個問題,如果用和我上述一樣用網頁模擬會發現網頁刷新是要等test.php執行完才會結束,也就是說如果我們把延時器的時間設成30分鐘會要等待30分鐘才會有反饋信息,這種方式肯定行不通的,所以需要使用異步訪問,比如使用web的ajax技術和其他異步技術,這里不再贅述
尾巴
以上只是我想到解決問題的想法和實施步驟,到了真正開發可能不會選擇這種方式,因為沒有經過性能測試,而且對于進程控制和線程控制并沒有多深入的了解,所以以后做訂單自動取消還是會選擇方法2的吧。
上述方法其實完全可以省掉Python那一步,我沒有去掉的原因是把我的實現經歷寫出來,因為我覺得開發期間可能真的會遇到這種多此一舉的方式,總之是要多思考,多看代碼,找出能優化的方案,這里感覺自己差得很遠,共勉吧
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:- PHP生成唯一訂單號的方法匯總
- php生成唯一的訂單函數分享
- PHP生成唯一訂單號
- PHP頁面跳轉實現延時跳轉的方法
- 用HTML/JS/PHP方式實現頁面延時跳轉的簡單實例