~~~
<?php
/**
* Created by PhpStorm.
* User: Mikkle
* QQ:776329498
* Date: 2017/06/19
* Time: 10:14
*/
namespace app\worker\controller;
use app\base\controller\Redis;
use think\Config;
use think\console\Command;
use think\console\Input;
use think\console\Output;
use think\Exception;
use think\Log;
/**
* Created by PhpStorm.
* Power by Mikkle
* QQ:776329498
* Date: 2017/6/12
* Time: 15:07
*/
abstract class WorkerCommandBase extends Command
{
protected $sleep = 5;
protected $redis;
protected $listName;
protected $pcntl;
public function __construct($name = null)
{
$this->redis = $this->redis();
$this->listName = "worker_list";
$this->pcntl = true;
parent::__construct($name);
}
protected function configure()
{
$this->setName('mikkle')->setDescription('Here is the mikkle\'s command ');
}
protected function execute(Input $input,Output $output)
{
while (true) {
//标记后端服务运行中
$this->signWorking();
echo "==================================================" . PHP_EOL;
$this->autoClass();
echo "==================================================" . PHP_EOL;
$this->sleep();
}
}
/**
* 自动执行
* Power: Mikkle
* Email:776329498@qq.com
* @return bool
*/
protected function autoClass()
{
$works = $this->getWorks();
if ($works) {
foreach ($works as $work => $item) {
if ($this->pcntl) {
$this->pcntlWorker($work, $item);
} else {
$this->runWorker($work, $item);
}
}
} else {
return false;
}
}
public function getWorks()
{
try {
return $this->redis->hget($this->listName);
} catch (Exception $e) {
return false;
}
}
/**
* 检测执行方法是否存在
* Power: Mikkle
* Email:776329498@qq.com
* @param $work
* @param $item
* @return bool
*/
protected function checkWorkerExists($work, $item)
{
if (class_exists($work)) {
if (method_exists($work, $item)) {
return true;
} else {
return false;
}
}
}
/**
* 运行任务
* Power: Mikkle
* Email:776329498@qq.com
* @param $work
* @param $item
*/
protected function runWorker($work, $item)
{
try {
if ($this->checkWorkerExists($work, $item)) {
echo "执行[{$work}]任务" . PHP_EOL;
$work::$item();
Log::notice("执行[{$work}::{$item}]任务");
} elseif ($this->checkWorkerExists($work, "run")) {
echo "执行[{$work}]任务" . PHP_EOL;
$work::run();
Log::notice("执行[{$work}::run]任务");
} else {
echo "执行[{$work}::{$item}]任务的默认和指定方法都不存在" . PHP_EOL;
$this->redis->hdel($this->listName, $item);
}
} catch (Exception $e) {
echo "执行[{$work}]任务失败" . PHP_EOL;
Log::notice($e->getMessage());
if ($this->pcntl) {
$this->pcntlKill();
}
}
}
/**
* 分进程
* Power: Mikkle
* Email:776329498@qq.com
* @param $work
* @param $item
*/
protected function pcntlWorker($work, $item)
{
try {
// 通过pcntl得到一个子进程的PID
$pid = pcntl_fork();
if ($pid == -1) {
// 错误处理:创建子进程失败时返回-1.
die ('could not fork');
} else if ($pid) {
// 父进程逻辑
// 等待子进程中断,防止子进程成为僵尸进程。
// WNOHANG为非阻塞进程,具体请查阅pcntl_wait PHP官方文档
pcntl_wait($status, WNOHANG);
} else {
// 子进程逻辑
$pid_2 = pcntl_fork();
if ($pid_2 == -1) {
// 错误处理:创建子进程失败时返回-1.
die ('could not fork');
} else if ($pid_2) {
// 父进程逻辑
echo "父进程逻辑开始" . PHP_EOL;
// 等待子进程中断,防止子进程成为僵尸进程。
// WNOHANG为非阻塞进程,具体请查阅pcntl_wait PHP官方文档
pcntl_wait($status, WNOHANG);
echo "父进程逻辑结束" . PHP_EOL;
} else {
// 子进程逻辑
echo "子进程逻辑开始" . PHP_EOL;
$this->runWorker($work, $item);
echo "子进程逻辑结束" . PHP_EOL;
$this->pcntlKill();
}
$this->pcntlKill();
}
} catch (Exception $e) {
Log::error($e->getMessage());
}
}
/**
* Kill子进程
* Power: Mikkle
* Email:776329498@qq.com
*/
protected function pcntlKill()
{
// 为避免僵尸进程,当子进程结束后,手动杀死进程
if (function_exists("posix_kill")) {
posix_kill(getmypid(), SIGTERM);
}
system('kill -9 ' . getmypid());
exit ();
}
public function signWorking()
{
$this->redis()->set("command", "true");
$this->redis()->setExpire("command", 30);
}
public function sleep($second = "")
{
$second = $second ? $second : $this->sleep;
// sleep(sleep($second)); //TP5的命令行 sleep($second) 不生效
sleep($second);
echo "睡眠{$second}秒成功!当前时间:" . date('h:i:s') . PHP_EOL;
}
/**
* @return int
*/
public function getSleep()
{
return $this->sleep;
}
/**
* @param int $sleep
* @return void
*/
public function setSleep($sleep)
{
$this->sleep = $sleep;
}
/**
* @title redis
* @description redis加载自定义Redis类
* User: Mikkle
* QQ:776329498
*/
protected static function redis()
{
$options = empty($options) ? $redis = Config::get("redis") : $options;
return Redis::instance($options);
}
}
~~~
- PHP7新特性
- 优雅的写代码
- 常见的代码优化
- 常用的工具类
- PHP原生生成EXCEL
- PHP地理位置计算
- PHP获取服务器状态
- 驼峰转下划线
- 百度地图两点坐标距离计算
- 判断是否是url
- PHP常见header头
- 邮件发送类
- 阿拉伯数字转化为大写
- 获取汉字首个拼音
- 根据身份证号获取星座
- 生成验证码类
- 生成唯一ID
- 身份证验证类
- PHP中文转拼音
- Nginx配置文件
- curl获取网页内容
- 快递查询api
- 上传图片类
- 股票类
- 找回密码类
- 字符串助手函数
- 校验数据规则
- PHP获取收集相关信息
- 字符串截取助手函数
- 网页中提取关键字
- 检测浏览器语言
- 微信相关类
- 微信获取access_token
- 获取用户基本信息
- 代码规范
- 编程规范(psr-1,2)
- 编程规范(原作者的建议)
- 经验
- 常用函数地址
- 函数集合
- 一些常识
- MYSQL相关知识
- 常用sql
- mysql事务隔离级别
- Read uncommitted
- Read committed
- Repeatable read
- Serializable
- 高性能MYSQL读书笔记
- 第一章MYSQL的架构
- mysql逻辑架构
- redis相关知识
- 1.安装redis
- 3.php操作redis
- 队列
- 悲观锁
- 乐观锁
- 发布
- 订阅
- redis实战-文章投票
- 设计模式
- 创建模型实例
- 单例模式
- 工厂模式
- AnimalInterface.php
- Chicken.php
- Factory.php
- Farm.php
- Pig
- SampleFactory.php
- Zoo
- 抽象工厂模式
- AnimalFactory
- Factory
- FarmInterface
- Income
- PandaZoo
- PeonyZoo
- PigFarm
- PlantFactory
- RiceFarm
- ZooInterface
- 原型模式
- 建造者模式
- 结构型模式实例
- 桥接模式
- 享元模式
- 外观模式
- 适配器模式
- 装饰器模式
- 组合模式
- 代理模式哦
- 过滤器模式
- 行为型模式实例
- 模板模式
- 策略模式
- 状态模式
- 观察者模式
- 责任链模式
- 访问者模式
- 解释器模式
- 空对象模式
- 中介者模式
- 迭代器模式
- 命令模式
- 备忘录模式
- 网络知识
- 互联网协议概述
- nginx简易交互过程
- HTTP知识
- LINUX相关知识
- swoole学习
- 1.初识swoole
- 2.WebSocket PHP 即时通讯开发
- 3.异步多进程的 CURL
- 4.异步非阻塞多进程的 Http 服务器
- 5.TCP 服务器
- 5.1同步 TCP 客户端
- 5.2异步 TCP 客户端
- 6.UDP 服务器
- 7.异步多任务处理
- 8.毫秒定时器
- 9.高并发投票
- ThinkPHP5学习
- 命令行操作
- 所有命令行中用到的基类
- 1.base
- 2.WorkerBase
- 3.TimeWorkerBase
- 4.CycleWorkerBase
- 5.WorkerCommandBase
- 6.WorkerHookBase
- 1.基础命令实现
- 2.建立Linux上的守护源码
- 3.发送模板消息
- 4.基于命令行实现自己的队列模式
- 5.发送定时短信
- thinkphp5使用sentry
- sentry通知,记录日志
- 高级查询
- Kafka相关
- 1.安装
- 2.为php打扩展
- 3.kafka实现
- 一些工具搭建
- sentry日志收集系统搭建
- walle搭建
- php实现定时任务
- 检测文件变化