php记录一次未支付订单过期处理的办法

订单是我们在日常开发中经常会遇到的一个功能,最近在做业务的时候需要实现客户下单之后订单超时未支付自动取消的功能,刚开始确认了几种方法:

  • 客户端到时间请求取消
  • 服务端定时查询有没有需要取消的订单,然后批量处理
  • 下单后创建定时器,延时处理
  • 使用redis或者memcache存储,设置过期时间,自动删除

上面的方法相信很多人都很熟悉了,虽然不熟悉也看过很多它的知识了吧,所以就不多废话了。今天要说的是第三种方法,为了防止进程进入阻塞,所以需要借助Swoole,workerman这样的框架来开启异步的定时器,今天讲解的是如何利用Swoole来处理订单过期时间。

环境要求

  • 安装swoole扩展 安装请参考swoole官网里面有详细的安装教程
  • 配置php.ini开启proc_open

swoole一次性定时器swoole_timer_after

官网介绍如下:

woole_timer_after函数是一个一次性定时器,执行完成后就会销毁。此函数与PHP标准库提供的sleep函数不同,after是非阻塞的。而sleep调用后会导致当前的进程进入阻塞,将无法处理新的请求。

执行成功返回定时器ID,若取消定时器,可调用 swoole_timer_clear

实现逻辑

image

完整代码

开启定时器

/*
 *开启定时器
 *id 订单id
 */
public function openTimer($id)
    {
        $arr = ['order_id' => $id];
        $json = base64_encode(json_encode($arr));
        //处理订单过期请求的地址
        $path = Config::get('order.past_order_url');
        // 定时器脚本路径
        $address = Config::get('order.timer_url');
        $cmd = "php Timer.php -a 1 -u {$path} -p {$json} -t 1800000";
        $descriptorspec = array(
            0 => ["pipe", "r"],
            1 => ["pipe", "w"],
            2 => ["file", "{$address}/error.txt", "a"]
        );
        proc_open($cmd, $descriptorspec, $pipes, $address, NULL);
        return true;
    }

Timer.php

class Timer
{
    public $request_url = null;

    public $params = [];

    public $time = 1000;

    public function __construct()
    {
        $param_arr = getopt('a:u:p:t:');
        if (!isset($param_arr['a'])) {
            $this->writeLog('【传参错误】操作类型不能为空', 2);
            die;
        }

        if (!isset($param_arr['u'])) {
            $this->writeLog('【传参错误】请求url不能为空', 2);
            die;
        }

        if ((isset($param_arr['t']) && $param_arr['t'] < 1000)) {
            $this->writeLog('【传参错误】时间不能小于1000毫秒', 2);
            die;
        }

        if (isset($param_arr['p']) && !is_string($param_arr['p'])) {
            $this->writeLog('【传参错误】请求参数必须是字符串', 2);
            die;
        }

        $this->request_url = $param_arr['u'];

        isset($param_arr['t']) && $this->time = $param_arr['t'];


        isset($param_arr['p']) && $this->params = ['data' => $param_arr['p']];

        if ($param_arr['a'] == 1) {
            $this->timer_after();
        }
    }

    /**
     * 一次性定时器
     * Created by 李果 En:AdoSir <1334435738@qq.com>
     * @return int
     */
    public function timer_after()
    {
        $id = swoole_timer_after($this->time, function () {
            //逻辑处理更具自己情况。我的解决方案是请求一个地址处理订单过期
            $result = $this->cu($this->request_url, json_encode(['data' => $this->params]));
            $this->writeLog("请求URL返回:code:{$result['httpCode']};response:{$result['response']}", 1);
        });
        $this->writeLog("添加定时器【id】:{$id}【参数】:" . json_encode($this->params), 1);
        return $id;
    }

    /**
     * 发起请求
     * @param $url
     * @param array $jsonStr
     * @return array
     */
    public function cu($url, $jsonStr = [])
    {
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_POST, 1);
        curl_setopt($ch, CURLOPT_URL, $url);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $jsonStr);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_HTTPHEADER, [
                'Content-Type: application/json; charset=utf-8',
                'Content-Length: ' . strlen($jsonStr),
            ]
        );
        $response = curl_exec($ch);
        $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
        curl_close($ch);

        return ['httpCode' => $httpCode, 'response' => $response];
    }

    /**
     * 写入日志
     * @param $msg
     * @param int $type
     */
    public function writeLog($msg, $type = 1)
    {
        //date_default_timezone_set("Asia/Shanghai");
        $day = date('Ymd');
        $file = $type == 1 ? "./log-{$day}.txt" : "./error-{$day}.txt";
        file_put_contents($file, "\r\n" . date("Y - m - d h:i:s") . ': ' . $msg, FILE_APPEND | LOCK_EX);
    }

}

new Timer();
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,539评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,594评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,871评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,963评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,984评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,763评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,468评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,357评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,850评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,002评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,144评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,823评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,483评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,026评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,150评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,415评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,092评论 2 355

推荐阅读更多精彩内容

  • swoole 安装用的是centOSphp安装目录:/usr/local/php php.ini配置文件路径:/u...
    a十二_4765阅读 11,849评论 3 9
  • 我家住在飞机场附近,天天看着飞机起飞和降落。我实在想亲身体会一次坐飞机的感受。 妈妈决定今年...
    静思_de8d阅读 311评论 0 0
  • 人类所有的仇恨,大概源自于欲望与贪婪 单纯善良一类的词,听起来真是讽刺 内心的黑暗,因贪婪和纵容而滋生,因欺骗与伤...
    神伤_难道阅读 529评论 0 0
  • 一 曾经有一段闭封的日子。 尝试过几个周都不跟别人交流。 感觉自己像个原始人,真的。看到以前很好的朋友,一起坐在操...
    木泽_阅读 528评论 0 1