#### 20180415 对源码进行了修正
~~~
<?php
/**
* Created by PhpStorm.
* User: Mikkle
* QQ:776329498
* Date: 2017/10/19
* Time: 10:10
*/
namespace mikkle\tp_worker;
use mikkle\tp_master\Db;
use mikkle\tp_master\Exception;
use mikkle\tp_master\Log;
use mikkle\tp_tools\Time;
abstract class CycleWorkBase
{
protected $listName;
protected $redis;
protected $workList;
protected $workerName;
public static $instance;
protected $connect =[] ;
protected $saveLog=false;
protected $tableName = "mk_log_service_queue";
protected $error;
protected $stopTime;
protected $startTime;
protected $nextTime=60;
/**
* Base constructor.
* @param array $options
*/
public function __construct($options=[])
{
$this->_initialize();
$this->redis = $this->redis();
$this->workList = "worker_list";
$this->workerName = get_called_class();
$this->listName = md5($this->workerName);
//校验时间
if (!is_numeric($this->startTime) && strtotime($this->startTime)) {
$this->startTime = strtotime($this->startTime);
}
if (!is_numeric($this->stopTime) && strtotime($this->stopTime)) {
$this->stopTime = strtotime($this->stopTime);
}
if (!empty($this->startTime ) && is_int( $this->startTime ) ){
if ( time()< $this->startTime ){
Log::notice("未到执行时间_".$this->startTime);
$this->sleep($this->startTime-time());
}
}
}
public function _initialize(){
}
/**
*
* 当命令行未运行 直接执行
* description add
* User: Mikkle
* QQ:776329498
* @param $data
* @return bool
*/
static public function start($data=["author"=>"mikkle"]){
try{
$data=json_encode($data);
$instance = static::instance();
self::clearWorkingStop();
switch (true){
case (self::checkWorking()):
Log::notice("Work service is Running!!");
return "CycleWork service is Running!!";
break;
case (self::checkCommandRun()):
$instance->redis->hSet("cycle_list",$instance->listName,$data);
Log::notice("Command service start work!!");
$instance->runWorker();
break;
default:
Log::notice("Command service No away!!");
return "Command service No away!!";
}
return "true";
}catch (Exception $e){
Log::error($e->getMessage());
return false;
}
}
/**
* title 循环处理的任务
* description runCycleHandle
* User: Mikkle
* QQ:776329498
* @param $data
* @return mixed
*/
abstract protected function runCycleHandle($data);
/**
* title 处理循环主逻辑 请勿修改
* description runHandle
* User: Mikkle
* QQ:776329498
* @param $data
*/
protected function runHandle($data)
{
try{
$instance=self::instance();
$i =0;
while ( true ){
echo self::checkWorkingStop() ;
if ( self::checkWorkingStop() ){
echo "任务被终止!";
break;
}
$i++;
self::signWorking();
$time = $instance->getNextRunTime();
if (!empty($instance->startTime ) && is_int( $instance->startTime ) ){
if ( time()> $instance->startTime ){
$instance->pcntlWorker($data,"runCycleHandle");
Log::notice("执行了程序pcntlWorker".$this->workerName );
}else{
Log::notice("未到执行时间_".$instance->startTime);
$this->sleep($instance->startTime-time());
continue;
}
}else{
$instance->pcntlWorker($data,"runCycleHandle");
Log::notice("执行了程序pcntlWorker".$this->workerName );
}
$this->sleep($time);
Log::notice("睡眠$time 秒,循环执行程序执行程序".$this->workerName );
echo "睡眠$time 秒,循环执行程序执行程序.$this->workerName ";
}
$this-> clearWorkingWork();
Log::notice("循环执行程序执结束");
}catch (Exception $e){
Log::notice( $e ->getMessage());
}
}
/**
*
* 当命令行未运行 直接执行
* description add
* User: Mikkle
* QQ:776329498
*/
static public function stop(){
return self::signWorkingStop();
}
/**
* title 查看运行状态
* description status
* User: Mikkle
* QQ:776329498
* @return bool
*/
static public function status(){
$instance=self::instance();
if (!empty($instance->startTime ) && is_int( $instance->startTime ) ){
if ( time()>strtotime( $instance->startTime )){
return (string)self::checkWorking();
}else{
return "waiting";
}
}else{
return (string)self::checkWorking();
}
}
protected function getNextRunTime(){
if (!empty( $this->nextTime ) && is_int( $this->nextTime)){
return $this->nextTime;
}
return 60;
}
/**
* @title runWorker
* @description 标注命令行执行此任务
* User: Mikkle
* QQ:776329498
* @param string $handleName
*/
protected function runWorker($handleName="run"){
$this->redis->hset($this->workList,$this->workerName,$handleName);
}
/**
* 标注命令行清除此任务
* Power: Mikkle
* Email:776329498@qq.com
*/
protected function clearWorker(){
$this->redis->hDel("cycle_list",$this->listName);
$this->redis->hdel($this->workList,$this->workerName);
}
/**
* Power: Mikkle
* Email:776329498@qq.com
* @return static
*/
static public function instance(){
if (self::$instance){
return self::$instance;
}
return new static();
}
/**
* 命令行执行的方法
* Power: Mikkle
* Email:776329498@qq.com
*/
static public function run(){
$instance = static::instance();
try {
$redisData = $instance->redis->hGet("cycle_list", $instance->listName);
if ($redisData) {
$data = json_decode($redisData, true);
if ($data) {
echo "开始执行循环任务" . PHP_EOL;
$instance->pcntlWorker($data);
$instance->redis->hDel("cycle_list", $instance->listName);
} else {
$instance->clearWorker();
}
} else {
$instance->clearWorker();
echo "未检测到循环任务" . PHP_EOL;
}
} catch (Exception $e) {
Log::error($e->getMessage());
echo($e->getMessage());
}
}
/**
* 检测命令行是否执行中
* Power: Mikkle
* Email:776329498@qq.com
* @return bool
*/
static public function checkCommandRun(){
return self::redis()->get("command") ? true :false;
}
public function getError(){
if (is_array($this->error )){
return json_encode( $this->error );
}
return $this->error;
}
/*
* 检查是注重某些值是非为空
*/
protected function checkArrayValueEmpty($array,$value,$error=true){
switch (true){
case (empty($array)||!is_array($array)):
if ($error==true){
$this->addError("要检测的数据不存在或者非数组");
}
return false;
break;
case (is_array($value)):
foreach ($value as $item){
if (!isset($array[$item]) || (empty($array[$item]) && $array[$item]!==0)){
if ($error==true) {
$this->addError("要检测的数组数据有不存在键值{$item}");
}
return false;
}
}
break;
case (is_string($value)):
if (!isset($array[$value]) || empty($array[$value] && $array[$value]!==0)){
if ($error==true) {
$this->addError("要检测的数组数据有不存在键值{$value}");
}
return false;
}
break;
default:
}
return true;
}
public function addError($error){
$this->error = is_string($error) ? $error : json_encode($error);
}
protected function saveRunLog($result,$data){
try{
$operateData = [
"class" => $this->workerName,
"args" => json_encode($data),
"result"=> $result ? "true":"false",
"error" => $this->error ? $this->getError() : null,
"time" => Time::getDefaultTimeString(),
];
Db::connect($this->connect)->table($this->tableName)->insert($operateData);
}catch (Exception $e){
Log::error($e->getMessage());
}
}
protected function sleep($time=1){
if ($time<=10){
sleep(sleep($time));
}else{
$ci = $time/10;
for($i=0; $i<$ci;$i++){
self::signWorking();
echo "已睡眠 10 秒";
sleep(10);
}
sleep(sleep($time%10));
}
}
/**
* title 分进程
* description pcntlWorker
* User: Mikkle
* QQ:776329498
* @param $data
* @param string $action
*/
protected function pcntlWorker($data,$action = "runHandle")
{
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->$action( $data );
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 ();
}
/**
* @title redis
* @description redis加载自定义Redis类
* User: Mikkle
* QQ:776329498
* @return \mikkle\tp_redis\Redis
*/
protected static function redis()
{
return WorkerRedis::instance();
}
static protected function signWorking($time=20){
self::redis()->set(self::instance()->workerName."_run","true",$time);
}
static protected function checkWorking(){
return self::redis()->get(self::instance()->workerName."_run") ? true :false;
}
static protected function clearWorkingWork(){
self::redis()->hDel("cycle_list",self::instance()->listName);
return self::redis()->delete( self::instance()->workerName."_stop");
}
static protected function signWorkingStop($time=3600*24){
self::redis()->hDel("cycle_list",self::instance()->listName);
return self::redis()->set(self::instance()->workerName."_stop","true",$time);
}
static protected function clearWorkingStop(){
return self::redis()->delete( self::instance()->workerName."_stop");
}
static protected function checkWorkingStop(){
if (!empty(self::instance()->stopTime )){
if ( time() > self::instance()->stopTime ){
return true;
}else{
return self::redis()->get(self::instance()->workerName."_stop") ? true :false;
}
}
return self::redis()->get(self::instance()->workerName."_stop") ? true :false;
}
}
~~~
- 序言及更新日志
- 前言一 开发PHP必备的环境(你可以不看)
- LinUX系统ThinkPHP5链接MsSQL数据库的pdo_dblib扩展
- centos7.2挂载硬盘攻略
- Centos系统Redis安装及Redis的PHP扩展安装
- Centos系统增加Swap(系统交换区)的方法
- 前言二 开发PHP软件配置和介绍(你依然可以不看)
- 数据库SQL文件
- 本地Git(版本控制)的搭建
- GIT远程仓库的克隆和推送
- Git常用命令
- PHP面向对象思想实战经验领悟
- PHP面向对象实战----命名空间
- PHP面向对象实战----继承
- 基类实战--底层方法封装
- 基类实战--构造函数实战
- 基类实战--析构函数的使用
- TP5实战开发前篇---控制器(controller)
- 控制器中Request类的使用
- 控制器中基类的使用
- TP5实战开发前篇---模型篇(model)
- TP5实战开发前篇---验证器篇(Validate)
- TP5实战课程入门篇---花拳绣腿
- 模块以及类的文件的建立
- Api开发------单条信息显示
- Api开发---单条信息复杂关联显示
- Api开发---查询信息缓存Cache的应用
- TP5实战技巧---开发思路 引路造桥
- TP5实战技巧---整合基类 化繁为简
- TP5实战课程入门篇---数据操作
- Api开发---数据的添加和修改
- API开发---快速开发API通用接口
- TP5专用微信sdk使用教程
- THINKPHP5微信SDK更新记录及升级指导
- TP5专用SDK 微信参数配置方法
- 微信公众号推送接口对接教程
- 微信推送接口对接示例含扫描登录微信端部分
- TP5专用微信支付SDK使用简介
- TP5专用支付宝支付SDK使用说明
- 使用NW将开发的网站打包成桌面应用
- TP5高阶实战课程 进阶篇概述
- 进阶篇一 实战开发之习惯及要求
- 进阶篇二 实战开发之控制器
- 控制器基类之控制器基类使用方法
- 控制器基类之控制器基类常用方法分享
- 控制器基类之构造函数的使用方法
- 进阶篇三 实战开发之权限控制
- TP5实战源码 --- 全局用户信息验证类Auth
- TP5实战源码 --- 微信Auth实战开发源码
- 进阶篇四 实战开发之模型
- 模型基类之模型基类的用途
- 模型基类之常用数据处理方法
- 模型逻辑层之实战代码(含事务)
- 模型实战开发之模型常用方法
- 模型实战源码 --- 乐观锁的应用
- 模型实战技巧---Model事件功能的使用
- 模型事件实战应用---数据库操作日志
- 进阶篇五 实战开发之缓存(Cache)
- TP5实战源码---应用缓存获取城市信息
- TP5实战源码---应用缓存获取分类详情
- 进阶篇六 TP5类库的封装和使用
- DataEdit快捷操作类库
- ShowCode快捷使用类库
- 阿里大于 短信API接口 TP5专用类库
- DatabaseUpgrade数据库对比及更新类库
- AuthWeb权限类使用说明
- 进阶篇七 服务层的应用
- 服务层源码示例
- 服务层基类源码
- 进阶篇八 应用层Redis数据处理基类
- Redis服务层基类源码
- 进阶篇九 使用Redis类库处理一般的抢购(秒杀)活动示例
- 进阶篇十 某大型项目应用本Redis类源码示例(含事务 乐观锁)
- 进阶篇十一 逻辑层的应用
- 逻辑层基类源码
- 进阶篇 服务层代码示例
- 高阶实战课程 进阶篇持续新增中
- 高阶篇一 TP5命令行之守护任务源码
- TP5实战源码 --- 命令行
- TP5实战源码 --- 通过shell建立PHP守护程序
- 高阶篇二 使用Redis队列发送微信模版消息
- 高阶篇二 之 Worker队列基类源码
- 高阶篇三 TP5实战之Redis缓存应用
- Redis实战源码之Hash专用类库源码
- Redis实战源码之Model类结合
- Redis实战源码之模型Hash基类源码
- Redis实战源码之Hash查询使用技巧
- Redis实战源码之 shell脚本中redis赋值和取值
- 高阶篇四 Swoole的实战应用
- swoole基类代码
- Swoole扩展WebsocketServer专用类
- 基于Swoole的多Room聊天室的程序
- Swoole守护服务shell源码
- 高阶篇五 命令行异步多进程队列类的应用
- tp_worker类源码
- WorkerBase
- WorkerCommand
- WorkerRedis
- Redis类
- CycleWorkBase
- WorkerHookBase异步钩子
- 队列日志SQL
- 高阶篇六 定时执行队列类库以及使用方法
- 定时队列类库源码
- 高阶篇七 异步执行循环队列类库以及使用教程
- CycleWorkBase源码
- 高阶实战课程 进阶篇持续新增中
- Extend便捷类库源码库
- 阿里相关类库
- SendSms--验证码API接口文件
- 权限相关类库目录
- AuthWeb 权限验证类库
- Redis便捷操作类库(20171224更新)
- Redis
- Tools工具类库集
- Curl类库
- DataEdit
- Rand类库
- ShowCode类库
- Upload类库
- 附件集合
- 附件一:微信支付 实战开发源码
- 微信支付类库源代码
- Common_util_pub.php
- DownloadBill_pub.php
- JsApi_pub.php
- NativeCall_pub.php
- NativeLink_pub.php
- OrderQuery_pub.php
- Refund_pub.php
- RefundQuery_pub.php
- SDKRuntimeException.php
- ShortUrl_pub.php
- UnifiedOrder_pub.php
- Wxpay_client_pub.php
- Wxpay_server_pub.php
- WxPayConf_pub.php
- 微信支付回调页面源码
- 附件二 顺丰快递BSP接口实战开发源码
- 顺丰快递BSP接口实战开发源码
- 顺丰BSP基类
- 顺丰BSP基础代码
- 顺丰BSP下单接口
- 顺丰BSP查单接口
- 顺丰BSP确认/取消接口
- 附件三 APP注册登陆接口源码(含融云平台接口)
- 附件四 TP5订单Model(含事务 获取器 修改器等方法)
- 附录五 RSA加密解密
- Rsa文件源码
- 附件六 阿里大于短信接口
- 附件七 AES加解密类
- AES加解密类源码
- 附件八 TP5路由设置源码
- 附件九 TP5 Excel导入导出下载便捷类库
- Excel类库TP5源码
- 附件十 TP5便捷操作Redis类库源码
- TP5源码 Redis操作便捷类库
- 附件十一 TP5源码 上传文件入库类源码
- 上传类Upload源码
- Upload类上传配置文件
- 存储图像文件的数据库SQL文件
- 存储文件的数据库SQL文件
- 附件十二 TP5 图片处理增强类 支持缩略图在线显示
- 附件十三 微信推送消息接口类库源码
- 附件十三 微信推送消息接口类库源码 之 基类
- 附件十四 存储微信昵称的处理方法