php swoole实现定时任务,Swoole实现任务定时自动化调度详解,来学习下

问题描述

这几天做银行对帐接口时,踩了一个坑,具体需求大致描述一下。

银行每天凌晨后,会开始准备昨天的交易流水数据,需要我们这边请求拿到。

因为他们给的是一个base64加密的zip压缩流,解开以后可以得到txt文件,里面就是我们需要的数据了。

业务程序写好以后,随手丢了一个定时任务就去睡觉了。

哪知道第二天上班的时候,检查。发现并没有拿到数据,查询一下日志的时候发现,凌晨服务端请求的时候,银行接口返回了:系统错误信息。

咨询银行那边后,银行那边相关人员建议我们多请求几次,但是在多次请求中,我发现银行那边是有频率限制的,最后得知,此接口只能半个小时才能请求一次。这就比较尴尬了,因为我不知道银行那边什么时候能返回数据给我。

于是这个问题怎么解决呢?理想的情况是,服务端请求数据,银行那边没有返回。然后程序等半个小时后,再请求一次,这样一直到银行那边返回正确的数据中止。

问题分析

这个功能换作别的语言也许不难,但是通过php实现的话,那就比较麻烦了。通常的话,我们可以搭配linux下的cron来实现,比如我们可以在凌晨到6:00之间做一个定时任务,每半个小时扫描一次php脚本,如果发现银行那边的状态依旧为失败的话,我们就执行一次php脚本去请求数据。直到请求到正确的数据,然后把状态更新为成功。

这不失为一种方法,但太傻了。比如说银行那边比较正常,凌晨,也就是第一次请求的时候,就已经返回了正确的数据,那么我们的cron脚本还傻傻的每个半个小时执行一次,好蠢!~

或者我们可以尝试使用linux下的at命令,但感觉还是不够优雅。

解决问题

于是决定给laravel扩展一个swoole插件来解决此问题,swoole的定时任务很完美的解决了我们目前的问题。

首先我们需要把swoole扩展安装好,具体过程略。

装好以后,我们写一个swoole简易的服务端测试脚本,注意,此脚本是放在app/Console/Commands/下的,笔者是放在了app/Console/Commands/Swoole/swoole.php下,具体代码为

namespace App\Console\Commands\Swoole;

use Illuminate\Console\Command;

class swoole extends Command

{

/**

* The name and signature of the console command.

*

* @var string

*/

protected $signature = 'swoole {action}';

/**

* The console command description.

*

* @var string

*/

protected $description = "Let's use swoole !";

private $serv;

/**

* Create a new command instance.

*

* @return void

*/

public function __construct()

{

parent::__construct();

}

/**

* Execute the console command.

*

* @return mixed

*/

public function handle()

{

$arg = $this->argument('action');

switch ($arg) {

case 'start':

$this->info('swoole server started');

$this->start();

break;

case 'stop':

$this->info('stoped');

$this->stop();

break;

case 'restart':

$this->info('restarted');

break;

}

}

private function start()

{

$this->serv = new \swoole_server("127.0.0.1", 9501);

$this->serv->set(array(

'worker_num' => 8,

'daemonize' => false,

'max_request' => 10000,

'dispatch_mode' => 2,

'task_worker_num' => 8,

'task_ipc_mode' => 3,

'log_file' => storage_path('logs/taskqueue.log'),

));

$this->serv->on('Receive', array($this, 'onReceive'));

$this->serv->on('Task', array($this, 'onTask'));

$this->serv->on('Finish', array($this, 'onFinish'));

$this->serv->start();

}

public function onReceive(\swoole_server $serv, $fd, $from_id, $data)

{

$serv->task($data);

}

public function onTask($serv, $task_id, $from_id, $data)

{

$timeon = (3) * 1000;

if ($timeon > 0) {

$serv->after($timeon, function () {

//业务逻辑处

exec('php /path/to/root/artisan Test:Command');

});

}

return date('Y-m-d H:i:s') . "第一次执行";

}

public function onFinish($serv, $task_id, $data)

{

echo "Task finish\n";

}

private function stop()

{

exec('/usr/bin/killall php');

}

}

这是服务端,我们主要用到了after方法,模拟的话,是三秒一执行。实际应该是三十分钟

然后我们随便写一个客户端连接类

/**

* Created by PhpStorm.

* User: nosay

* Date: 4/13/18

* Time: 9:27 PM

*/

namespace App\Extension\php\Swoole;

class swoole{

private $data;

private $client;

public function __construct($data){

$this->data = $data;

$this->client = new \swoole_client(SWOOLE_SOCK_TCP);

}

public function connect(){

if( !$this->client->connect("127.0.0.1", 9501 , 1) ) {

echo "Error";

}

$this->client->send($this->data);

}

}

于是我们在银行脚本中就可以去执行了

namespace App\Console\Commands\Test;

use App\Extension\php\Swoole\swoole;

use Illuminate\Console\Command;

class TestCommand extends Command

{

/**

* The name and signature of the console command.

*

* @var string

*/

protected $signature = 'Test:Command';

/**

* The console command description.

*

* @var string

*/

protected $description = 'Command Test';

/**

* Create a new command instance.

*

* @return void

*/

public function __construct()

{

parent::__construct();

}

/**

* Execute the console command.

*

* @return mixed

*/

public function handle()

{

//这里是业务逻辑

//如果银行那边返回的为false的话,那么我们把他交给swoole的定时脚本

$status = false;

if(!$status)

{

$swoole = new swoole("hehe");

$swoole->connect();

}

}

}

以上内容希望帮助到大家,很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值