主要原理是通过PHP创建多个子进程,在子进程中发送进程闹钟信号,然后再监听闹钟信号中继续发送闹钟信号。同时通过父进程设置非阻塞运行。代码如下:
<?php
/**
* 订单任务
*/
class Order
{
public function run()
{
file_put_contents("log1.txt", '我是订单任务,本次运行时间:' . date('Y-m-d H:i:s') . "\r\n", FILE_APPEND);
}
}
/**
* 商品任务
*/
class Goods
{
public function run()
{
file_put_contents("log2.txt", '我是商品任务,本次运行时间:' . date('Y-m-d H:i:s') . "\r\n", FILE_APPEND);
}
}
/**
* Task任务核心
*/
class Console
{
/**
* 定时任务配置和执行时间间隔
*/
protected static $tasks = [
'Order' => 5,
'Goods' => 10,
];
/**
* 初始化为每个任务分配进程
*/
public static function init()
{
$tasks = self::$tasks;
foreach ($tasks as $task => $second)
{
//创建子进程
$pid = pcntl_fork();
if ($pid == -1)
{
die ('创建子进程失败!');
}
elseif ($pid)
{
//等待子进程执行完成回收,并且设置非阻塞
pcntl_wait($status, WNOHANG);
}
else
{
self::process($task, $second);
//子进程执行完成关闭,避免重复fork
die();
}
}
}
/**
* 子进程闹钟处理任务
*/
public static function process($task, $second)
{
//安装信号管理
pcntl_signal(SIGALRM, function () use ($task, $second) {
pcntl_alarm($second);
$taskObj = new $task();
call_user_func([$taskObj, 'run']);
}, false);
//发送闹钟信号
pcntl_alarm($second);
//调用等待信号的处理器
while (true)
{
pcntl_signal_dispatch();
}
}
}
//初始化
Console::init();程序执行后.主进程会根据已经配置的任务数来分配子进程,每个子进程来定时执行对应的任务,可以用来做redis消费者。
以上只是给你一个思路来实现,你还可以直接用我开源的定时器直接使用,访问地址:https://gitee.com/392223903/EasyTask
先看看下面的代码:<?php session_start(); $_SESSION['username']='lucy'; ?>当我们请求访问上面的脚本,默认会在我们的客户端生成一个名为PHPSESSID的cookie,我这里的值是PHPSESSID=...
if($_SERVER['REQUEST_METHOD'] == 'POST') { echo('This is post '); } elseif ($_SERVER['...
项目中使用服务来执行webservice,由于对方系统api不稳定,经常导致服务崩溃,只能重启,一个月差不多要重启一次。初期的解决办法是捕获异常,然后continue掉。<?php try { $url = 'http...
Redis提供了发布订阅功能,可以用于消息的传输,Redis的发布订阅机制包括三个部分,发布者(publisher),订阅者(subscriber)和频道(channel)。 发布者和订阅者都是Redis客户端,Channel则为Redis服务器端,发布者将消息发送到某个的频道,订阅了这个...
在使用ftok生成ipc进程通信key尝试将第二个参数项目标识符传入字符串报错:PHP Warning: ftok(): Project identifier is invalid,查阅资料发现第二个字符串只能是1个字符串,长度为1....
第一次听说端口复用是在mixphp最新版本中发现的,mixphp启动监听9501端口,现在作者说可以多开几个进程来执行mixphp,我心里想了下再启动不是会端口冲突嘛,但是却没有问题,于是下载mixphp的源码解读,原来是启动http服务器使用new Co\Http\Server('0.0....