最近在做業務的時候需要實現客戶下單之後訂單超時未支付自動取消的功能,剛開始確認了幾種方法:客戶端到時間請求取消服務端定時查詢有沒有需要取消的訂單,然後批次處理,下單後創建定時器,延時處理使用 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); }
##然後在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/... 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 "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
那一步,直接用 PHP
的 exec
函数来执行 PHP
文件。
测试通过后发现这种方法是可以创建定时器并且通过web远程使用的,不过有个问题,如果用和我上述一样用网页模拟会发现网页刷新是要等 test.php
执行完才会结束,也就是说如果我们把延时器的时间设成30分钟会要等待30分钟才会有反馈信息,这种方式肯定行不通的,所以需要使用异步访问,比如使用web的 ajax
技术和其他异步技术,这里不再赘述
以上只是我想到解决问题的想法和实施步骤,到了真正开发可能不会选择这种方式,因为没有经过性能测试,而且对于进程控制和线程控制并没有多深入的了解,所以以后做订单自动取消还是会选择方法2的吧。
上述方法其實完全可以省掉 Python
那一步,我沒有去掉的原因是把我的實現經驗寫出來,因為我覺得開發期間可能真的會遇到這種多此一舉的方式,總之是要多思考,多看程式碼,找出能優化的方案,這裡感覺自己差得很遠,共勉吧
以上是PHP實作訂單延時處理的方法實例的詳細內容。更多資訊請關注PHP中文網其他相關文章!