由于项目中再修改产品的同时要同步关联水单,删单,客保 等等数据。所以不可能等待所有都执行完毕以后再给客户端反馈。所以查阅资料,自己用tp框架写了个队列。在这里晒出来代码,以供大家参考。
思路
1,需要用到队列则扔到queue表中。
2,利用linux计划任务 * * * * * 去执行 CronMission方法
3利用 flock文件排它锁,保证单线程执行,避免重复执行
具体实现如下:
表:
CREATE TABLE IF NOT EXISTS `roav2_queue` (
`id` int(10) NOT NULL AUTO_INCREMENT,) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
1,获取队列
public function getQueueTask($limit = 10) { $map = ['status' => 0]; $data = $this->where($map)->order('id asc')->limit($limit)->select(); // var_dump(M()->getLastSql());die; return $data; }
2,参数拼接
public function cmdHandle($phpCmd, $taskPath, $param) { $str = ''; if($param) { $params = unserialize($param); foreach ($params as $k => $v) { $str .= '/' . $k . '/' . $v; } $taskPath .= $str; } $res = $phpCmd . $taskPath; return $res; }
3,修改任务状态
public function changeTaskStatus($id, $status) { return save_r($this, ['id' => $id], ['status' => $status]); }
4,执行队列
public function doQueue() { $phpCmd = C('PHPCMD'); $logPath = C('CRON_LOG_PATH'); $fp = fopen('flock.txt', 'w+'); if (flock($fp, LOCK_EX|LOCK_NB))//使用文件排他锁保证单线程执行 { $tasks = $this->getQueueTask(10); foreach ($tasks as $v) { $job = $this->cmdHandle($phpCmd, $v['taskpath'], $v['param']); $job .= ' >> ' . $logPath . 'doQueueMission' . date('Y-m-d') . '.log '; system($job); $this->changeTaskStatus($v['id'], 1); } flock($fp, LOCK_UN); } fclose($fp); }
5,添加任务
public function addTask($taskPath, $param) { if(empty($taskPath) || empty($param)) return 0; $data = [ 'taskpath' => $taskPath, 'param' => serialize($param), 'created_at' => date('Y-m-d H:i:s') ]; $res = add_r($this, $data); if ($res) return 1; else return 0; }
6,计划任务
public function cronMission() { $minute = date('i'); //5分钟执行一次 if ($minute%5 == 0) { $this->model->doQueue(); } }大概就这些了,有任何问题,希望大家及时和我反馈哈。