首頁 後端開發 php教程 php 守護程式(Daemon)

php 守護程式(Daemon)

May 28, 2018 pm 03:49 PM
php 行程

這篇文章給大家分享的內容是關於php 守護進程(Daemon),有著一定的參考價值,有需要的朋友可以參考一下

 守護程式(Daemon)是運行在後台的一種特殊進程。它獨立於控制終端並且週期性地執行某種任務或等待處理某些發生的事件。守護程式是一種很有用的程式。 php也可以實現守護程式的功能。

 

1、基本概念

    程序

            每個行程有一個父程式,子行程退出,父程式能由子程式退出的狀態。

    進程組

            每個行程都屬於一個進程組,且每個進程組都有一個進程組號,該號等於該進程組組長的PID

2.守護程式要點

    1. 在背景運作。    
         為避免以暫停控制終端將Daemon放入後台執行。方法是在進程中呼叫fork使父進程終止,讓Daemon在子進程中後台執行。 if($pid=pcntl_fork()) exit(0);//是父進程,結束父進程,子進程繼續
    2. 脫離控制終端,登入會話和進程組 
       中的進程與控制終端,登入會話和進程組之間的關係:進程屬於一個進程組,進程組號(GID)就是進程組長的進程號(PID)。登入會話可以包含多個進程組。這些進程組共用一個控制終端。這個控制終端通常是創建進程的登入終  端。控制終端,登入會話和進程組通常是從父進程繼承下來的。我們的目的就是要擺脫它們,使之不受它們​​的影響。方法是在第1點的基礎上,呼叫setsid() 使進程成為會話組長: posix_setsid();
        說明:當進程是會話組長時setsid()呼叫失敗。但第一點已經保證進程不是會話組長。 setsid()呼叫成功後,進程成為新的會話組長和新的進程組長,並與原先的登入會話和進程組脫離。由於會話過程對控制終端的獨佔性,進程同時與控制終端脫離。
    3. 禁止程序重新開啟控制終端機
        現在,且已成為無終端機的會話組長。但它可以重新申請打開一個控制終端。可以透過讓進程不再成為會話群組長度來禁止進程重新開啟控制終端: if($pid=pcntl_fork()) exit(0);//結束第一子進程,第二子進程繼續(第二子程序不再是會話組長)
    4. 關閉開啟的檔案描述子
        以建立它的父行程繼承了開啟的檔案描述子。如不關閉,將會浪費系統資源,造成進程所在的檔案系統無法卸下以及造成無法預料的錯誤。依下列方法關閉它們:
        fclose(STDIN),fclose(STDOUT),fclose(STDERR)關閉標準輸入輸出與錯誤顯示。
    5. 改變目前工作目錄
        處理程序活動時,其工作目錄所在的檔案系統無法移除。一般需要將工作目錄改變到根目錄。對於需要轉儲核心,寫入運行日誌的程序將工作目錄改變到特定目錄如chdir("/")
    6. 重設文件創建掩模
        進程從創建它的父進程那裡繼承了文件創建掩模。它可能修改守護程式所建立的檔案的存取位。為防止這一點,將檔案建立遮罩清除:umask(0);
    7.處理SIGCHLD訊號
        處理SIGCHLD訊號並不是必須的。但對於某些進程,特別是伺服器進程往往在請求到來時產生子進程處理請求。如果父行程不等待子行程結束,子行程將成為殭屍行程(zombie)從而佔用系統資源。如果父進程等待子進程結束,將增加父進程的負擔,影  響伺服器進程的並發效能。在Linux下可以簡單地將SIGCHLD訊號的操作設為SIG_IGN。 signal(SIGCHLD,SIG_IGN);
        這樣,核心在子行程結束時不會產生殭屍行程。這點與BSD4不同,BSD4下必須明確等待子程序結束才能釋放殭屍行程。關於訊號的問題請參考Linux 訊號說明清單

  1. <?php  
    /** 
    *@author tengzhaorong@gmail.com 
    *@date 2013-07-25 
    * 后台脚本控制类 
    */  
    class DaemonCommand{  
       
        private $info_dir="/tmp";  
        private $pid_file="";  
        private $terminate=false; //是否中断  
        private $workers_count=0;  
        private $gc_enabled=null;  
        private $workers_max=8; //最多运行8个进程  
       
        public function __construct($is_sington=false,$user=&#39;nobody&#39;,$output="/dev/null"){  
       
                $this->is_sington=$is_sington; //是否单例运行,单例运行会在tmp目录下建立一个唯一的PID  
                $this->user=$user;//设置运行的用户 默认情况下nobody  
                $this->output=$output; //设置输出的地方  
                $this->checkPcntl();  
        }  
        //检查环境是否支持pcntl支持  
        public function checkPcntl(){  
            if ( ! function_exists(&#39;pcntl_signal_dispatch&#39;)) {  
                // PHP < 5.3 uses ticks to handle signals instead of pcntl_signal_dispatch  
                // call sighandler only every 10 ticks  
                declare(ticks = 10);  
            }  
       
            // Make sure PHP has support for pcntl  
            if ( ! function_exists(&#39;pcntl_signal&#39;)) {  
                $message = &#39;PHP does not appear to be compiled with the PCNTL extension.  This is neccesary for daemonization&#39;;  
                $this->_log($message);  
                throw new Exception($message);  
            }  
            //信号处理  
            pcntl_signal(SIGTERM, array(__CLASS__, "signalHandler"),false);  
            pcntl_signal(SIGINT, array(__CLASS__, "signalHandler"),false);  
            pcntl_signal(SIGQUIT, array(__CLASS__, "signalHandler"),false);  
       
            // Enable PHP 5.3 garbage collection  
            if (function_exists(&#39;gc_enable&#39;))  
            {  
                gc_enable();  
                $this->gc_enabled = gc_enabled();  
            }  
        }  
       
        // daemon化程序  
        public function daemonize(){  
       
            global $stdin, $stdout, $stderr;  
            global $argv;  
       
            set_time_limit(0);  
       
            // 只允许在cli下面运行  
            if (php_sapi_name() != "cli"){  
                die("only run in command line mode\n");  
            }  
       
            // 只能单例运行  
            if ($this->is_sington==true){  
       
                $this->pid_file = $this->info_dir . "/" .__CLASS__ . "_" . substr(basename($argv[0]), 0, -4) . ".pid";  
                $this->checkPidfile();  
            }  
       
            umask(0); //把文件掩码清0  
       
            if (pcntl_fork() != 0){ //是父进程,父进程退出  
                exit();  
            }  
       
            posix_setsid();//设置新会话组长,脱离终端  
       
            if (pcntl_fork() != 0){ //是第一子进程,结束第一子进程     
                exit();  
            }  
       
            chdir("/"); //改变工作目录  
       
            $this->setUser($this->user) or die("cannot change owner");  
       
            //关闭打开的文件描述符  
            fclose(STDIN);  
            fclose(STDOUT);  
            fclose(STDERR);  
       
            $stdin  = fopen($this->output, &#39;r&#39;);  
            $stdout = fopen($this->output, &#39;a&#39;);  
            $stderr = fopen($this->output, &#39;a&#39;);  
       
            if ($this->is_sington==true){  
                $this->createPidfile();  
            }  
       
        }  
        //--检测pid是否已经存在  
        public function checkPidfile(){  
       
            if (!file_exists($this->pid_file)){  
                return true;  
            }  
            $pid = file_get_contents($this->pid_file);  
            $pid = intval($pid);  
            if ($pid > 0 && posix_kill($pid, 0)){  
                $this->_log("the daemon process is already started");  
            }  
            else {  
                $this->_log("the daemon proces end abnormally, please check pidfile " . $this->pid_file);  
            }  
            exit(1);  
       
        }  
        //----创建pid  
        public function createPidfile(){  
       
            if (!is_dir($this->info_dir)){  
                mkdir($this->info_dir);  
            }  
            $fp = fopen($this->pid_file, &#39;w&#39;) or die("cannot create pid file");  
            fwrite($fp, posix_getpid());  
            fclose($fp);  
            $this->_log("create pid file " . $this->pid_file);  
        }  
       
        //设置运行的用户  
        public function setUser($name){  
       
            $result = false;  
            if (empty($name)){  
                return true;  
            }  
            $user = posix_getpwnam($name);  
            if ($user) {  
                $uid = $user[&#39;uid&#39;];  
                $gid = $user[&#39;gid&#39;];  
                $result = posix_setuid($uid);  
                posix_setgid($gid);  
            }  
            return $result;  
       
        }  
        //信号处理函数  
        public function signalHandler($signo){  
       
            switch($signo){  
       
                //用户自定义信号  
                case SIGUSR1: //busy  
                if ($this->workers_count < $this->workers_max){  
                    $pid = pcntl_fork();  
                    if ($pid > 0){  
                        $this->workers_count ++;  
                    }  
                }  
                break;  
                //子进程结束信号  
                case SIGCHLD:  
                    while(($pid=pcntl_waitpid(-1, $status, WNOHANG)) > 0){  
                        $this->workers_count --;  
                    }  
                break;  
                //中断进程  
                case SIGTERM:  
                case SIGHUP:  
                case SIGQUIT:  
       
                    $this->terminate = true;  
                break;  
                default:  
                return false;  
            }  
       
        }  
        /** 
        *开始开启进程 
        *$count 准备开启的进程数 
        */  
        public function start($count=1){  
       
            $this->_log("daemon process is running now");  
            pcntl_signal(SIGCHLD, array(__CLASS__, "signalHandler"),false); // if worker die, minus children num  
            while (true) {  
                if (function_exists(&#39;pcntl_signal_dispatch&#39;)){  
       
                    pcntl_signal_dispatch();  
                }  
       
                if ($this->terminate){  
                    break;  
                }  
                $pid=-1;  
                if($this->workers_count<$count){  
       
                    $pid=pcntl_fork();  
                }  
       
                if($pid>0){  
       
                    $this->workers_count++;  
       
                }elseif($pid==0){  
       
                    // 这个符号表示恢复系统对信号的默认处理  
                    pcntl_signal(SIGTERM, SIG_DFL);  
                    pcntl_signal(SIGCHLD, SIG_DFL);  
                    if(!empty($this->jobs)){  
                        while($this->jobs[&#39;runtime&#39;]){  
                            if(empty($this->jobs[&#39;argv&#39;])){  
                                call_user_func($this->jobs[&#39;function&#39;],$this->jobs[&#39;argv&#39;]);  
                            }else{  
                                call_user_func($this->jobs[&#39;function&#39;]);  
                            }  
                            $this->jobs[&#39;runtime&#39;]--;  
                            sleep(2);  
                        }  
                        exit();  
       
                    }  
                    return;  
       
                }else{  
       
                    sleep(2);  
                }  
       
       
            }  
       
            $this->mainQuit();  
            exit(0);  
       
        }  
       
        //整个进程退出  
        public function mainQuit(){  
       
            if (file_exists($this->pid_file)){  
                unlink($this->pid_file);  
                $this->_log("delete pid file " . $this->pid_file);  
            }  
            $this->_log("daemon process exit now");  
            posix_kill(0, SIGKILL);  
            exit(0);  
        }  
       
        // 添加工作实例,目前只支持单个job工作  
        public function setJobs($jobs=array()){  
       
            if(!isset($jobs[&#39;argv&#39;])||empty($jobs[&#39;argv&#39;])){  
       
                $jobs[&#39;argv&#39;]="";  
       
            }  
            if(!isset($jobs[&#39;runtime&#39;])||empty($jobs[&#39;runtime&#39;])){  
       
                $jobs[&#39;runtime&#39;]=1;  
       
            }  
       
            if(!isset($jobs[&#39;function&#39;])||empty($jobs[&#39;function&#39;])){  
       
                $this->log("你必须添加运行的函数!");  
            }  
       
            $this->jobs=$jobs;  
       
        }  
        //日志处理  
        private  function _log($message){  
            printf("%s\t%d\t%d\t%s\n", date("c"), posix_getpid(), posix_getppid(), $message);  
        }  
       
    }  
       
    //调用方法1  
    $daemon=new DaemonCommand(true);  
    $daemon->daemonize();  
    $daemon->start(2);//开启2个子进程工作  
    work();  
       
       
       
       
    //调用方法2  
    $daemon=new DaemonCommand(true);  
    $daemon->daemonize();  
    $daemon->addJobs(array(&#39;function&#39;=>&#39;work&#39;,&#39;argv&#39;=>&#39;&#39;,&#39;runtime&#39;=>1000));//function 要运行的函数,argv运行函数的参数,runtime运行的次数  
    $daemon->start(2);//开启2个子进程工作  
       
    //具体功能的实现  
    function work(){  
          echo "测试1";  
    }  
    ?>
    登入後複製

相關推薦:

PHP進程間通訊詳解

總結關於PHP進程通訊注意點


#

以上是php 守護程式(Daemon)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

我後悔之前不知道的 7 個 PHP 函數 我後悔之前不知道的 7 個 PHP 函數 Nov 13, 2024 am 09:42 AM

如果您是經驗豐富的PHP 開發人員,您可能會感覺您已經在那裡並且已經完成了。操作

如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

您如何在PHP中解析和處理HTML/XML? 您如何在PHP中解析和處理HTML/XML? Feb 07, 2025 am 11:57 AM

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

See all articles