業(yè)務(wù)需求
訂單是我們?cè)谌粘i_發(fā)中經(jīng)常會(huì)遇到的一個(gè)功能,最近在做業(yè)務(wù)的時(shí)候需要實(shí)現(xiàn)客戶下單之后訂單超時(shí)未支付自動(dòng)取消的功能,剛開始確認(rèn)了幾種方法:
- 客戶端到時(shí)間請(qǐng)求取消
- 服務(wù)端定時(shí)查詢有沒有需要取消的訂單,然后批量處理
- 下單后創(chuàng)建定時(shí)器,延時(shí)處理
- 使用redis或者memcache存儲(chǔ),設(shè)置過期時(shí)間,自動(dòng)刪除
綜合考慮上述方法,第一種最先排除,因?yàn)槿绻蛻舭袮PP后臺(tái)禁止或者網(wǎng)絡(luò)連接禁止,那么就無法發(fā)給服務(wù)端請(qǐng)求,訂單就會(huì)一直是未處理狀態(tài);第二種方法使用的比較多,不過存在準(zhǔn)確度的問題,還有需要確認(rèn)定時(shí)任務(wù)的周期,暫時(shí)列為后補(bǔ)方法;第四種方法存在的問題就是訂單如果刪除就是物理刪除,無法統(tǒng)計(jì)未處理數(shù)據(jù)(當(dāng)然可以存redis時(shí)候順便存在mysql這樣的數(shù)據(jù)庫做長(zhǎng)久存儲(chǔ)然后用方法二定時(shí)處理)。
最終準(zhǔn)備使用方法三。
再確認(rèn)使用方法3的時(shí)候,由于使用的PHP這種開發(fā)語言,所以想實(shí)現(xiàn)定時(shí)器功能需要借助Swoole或者workerman。由于Swoole是C開發(fā)的擴(kuò)展框架,性能方面肯定比較好,就選了Swoole。
前期準(zhǔn)備
- 使用Swoole首先需要在服務(wù)器上安裝Swoole擴(kuò)展,安裝方法和安裝其他擴(kuò)展大同小異,可以參考這邊文章
- 安裝完之后檢測(cè)下擴(kuò)展是否正常安裝,查看phpinfo或者PHP-m,如果出現(xiàn)Swoole,則說明安裝成功
- Swoole官方文檔有定時(shí)器的相關(guān)文檔
開始測(cè)試
我們創(chuàng)建一個(gè)swoole_test.php文件和一個(gè)log.txt文件(用來測(cè)試),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);
}
然后在網(wǎng)頁訪問這個(gè)PHP文件,結(jié)果如下:
然后在Linux終端運(yùn)行PHP:/usr/local/php7/bin/php /home/app/swoole_test.php,結(jié)果如下:
內(nèi)心一陣。。。
原來定時(shí)器只能在cli模式下,那么這個(gè)想法怕是要GG了,難道就栽倒這里了嗎,難道就沒有別的方法了嗎?就在我欲哭無淚的時(shí)候突然靈光乍現(xiàn),一個(gè)詞閃到我的腦海:Python!
對(duì),我們不能單單靠著PHP啊,還有Python這種神奇的語言呢,我們知道Python的os模塊里的os.system方法是可以執(zhí)行命令行的,那么不就可以實(shí)現(xiàn)在cli模式下運(yùn)行剛才的swoole_test.php文件了么。
內(nèi)心一陣激動(dòng)后,覺得測(cè)試是否可行
我們知道Linux都是自帶Python的,但是不同的版本Python版本不同,有的自帶的是Python2.6,版本過低了,所以需要裝一個(gè)高版本的,這里我選擇Python3,注意不要覆蓋系統(tǒng)自帶的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,如果出現(xiàn)
則安裝成功。
安裝完P(guān)ython3之后,我們新建一個(gè)test.py文件,內(nèi)容如下:
#!usr/bin/env python3`
#-*- coding:utf-8 -*-
import os
ret = os.system("/usr/local/php7/bin/php /home/app/swoole_test.php") #請(qǐng)使用自己系統(tǒng)的絕對(duì)路徑
print(ret)
然后我們?cè)诮K端執(zhí)行:/usr/bin/python3 /home/app/test.py,注意:這里只是執(zhí)行PHP文件,但是文件里的echo內(nèi)容是不會(huì)在終端輸出的,這時(shí)候就用到剛才新建的log.txt文件了。執(zhí)行完P(guān)ython文件后,我們?nèi)og文件檢查下,發(fā)現(xiàn)內(nèi)容已經(jīng)寫入,所以使用Python是可以實(shí)現(xiàn)PHP的cli模式的。┗|`O′|┛ 嗷~~
到這里就會(huì)有同學(xué)疑惑了,你這使用Python實(shí)現(xiàn)了PHP的cli模式,但是怎么通過web遠(yuǎn)程訪問呢?這個(gè)時(shí)候就用到PHP的exec方法了,我們知道PHP的exec方法和Python的os.system方法一樣是可以執(zhí)行命令行命令的,所以我們可以新建一個(gè)test.php文件,內(nèi)容如下:
?php
$program="/usr/bin/python3 /home/app/nongyephp/test.py"; #注意使用絕對(duì)路徑
echo "beginbr>";
(exec ($program));
echo "endbr>";
die;
然后我們通過網(wǎng)頁訪問test.php文件。結(jié)果如下:
然后去log文件檢查,發(fā)現(xiàn)也寫入日志了,所以這個(gè)方法是可行的!
做到這里心里美滋滋的,不過老覺得好像哪里不對(duì),終于終于意識(shí)到一個(gè)很傻逼的問題:既然PHP可以直接有命令行函數(shù),為啥多此一舉借助Python然后在用Python的函數(shù)呢?這不是脫了褲子放屁多此一舉嗎?
再大罵自己是傻逼N遍之后,我默默修改了test.php文件內(nèi)容:
?php
echo "beginbr>";
$program="/usr/local/php7/bin/php /home/app/nongyephp/swoole_test.php"; #注意使用絕對(duì)路徑
(exec ($program));
echo "endbr>";
die;
在直接訪問test.php文件,反饋結(jié)果和借助Python一樣,這樣就可以免去Python那一步,直接用PHP的exec函數(shù)來執(zhí)行PHP文件。
結(jié)尾
測(cè)試通過后發(fā)現(xiàn)這種方法是可以創(chuàng)建定時(shí)器并且通過web遠(yuǎn)程使用的,不過有個(gè)問題,如果用和我上述一樣用網(wǎng)頁模擬會(huì)發(fā)現(xiàn)網(wǎng)頁刷新是要等test.php執(zhí)行完才會(huì)結(jié)束,也就是說如果我們把延時(shí)器的時(shí)間設(shè)成30分鐘會(huì)要等待30分鐘才會(huì)有反饋信息,這種方式肯定行不通的,所以需要使用異步訪問,比如使用web的ajax技術(shù)和其他異步技術(shù),這里不再贅述
尾巴
以上只是我想到解決問題的想法和實(shí)施步驟,到了真正開發(fā)可能不會(huì)選擇這種方式,因?yàn)闆]有經(jīng)過性能測(cè)試,而且對(duì)于進(jìn)程控制和線程控制并沒有多深入的了解,所以以后做訂單自動(dòng)取消還是會(huì)選擇方法2的吧。
上述方法其實(shí)完全可以省掉Python那一步,我沒有去掉的原因是把我的實(shí)現(xiàn)經(jīng)歷寫出來,因?yàn)槲矣X得開發(fā)期間可能真的會(huì)遇到這種多此一舉的方式,總之是要多思考,多看代碼,找出能優(yōu)化的方案,這里感覺自己差得很遠(yuǎn),共勉吧
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
您可能感興趣的文章:- PHP生成唯一訂單號(hào)的方法匯總
- php生成唯一的訂單函數(shù)分享
- PHP生成唯一訂單號(hào)
- PHP頁面跳轉(zhuǎn)實(shí)現(xiàn)延時(shí)跳轉(zhuǎn)的方法
- 用HTML/JS/PHP方式實(shí)現(xiàn)頁面延時(shí)跳轉(zhuǎn)的簡(jiǎn)單實(shí)例