<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Nginx的進程管理與重載原理

          共 7238字,需瀏覽 15分鐘

           ·

          2021-04-25 10:03

          點擊上方藍色字體,選擇“標星公眾號”

          優(yōu)質(zhì)文章,第一時間送達

            作者 |  itbsl

          來源 |  urlify.cn/ZRZZVz

          進程結(jié)構(gòu)圖

          Nginx是多進程結(jié)構(gòu),多進程結(jié)構(gòu)設(shè)計是為了保證Nginx的高可用高可靠,包含:

          • master進程:父進程,負責worker進程的管理

          • worker進程:子進程,worker進程一般配置與服務(wù)器CPU核數(shù)相同,worker進程用來處理具體請求。

          • cache進程:也是子進程,包括cache manager和cache loader進程,主要是反向代理做緩存使用。

          注:多進程相對于多線程之所以能夠保證高可用與高可靠是因為進程間地址空間是獨立的,進程間的任務(wù)不會相互影響,相對多線程更加耗費CPU資源。而多線程共享一個進程的地址空間,其中一個線程任務(wù)失敗會影響到其它線程任務(wù)。

          圖3-1 Nginx進程結(jié)構(gòu)圖

          假設(shè)我們的Nginx服務(wù)的用戶是nginx,我們可以使用如下命令查看當前運行的Nginx服務(wù)的master進程和worker進程,而且可以看到4個worker進程的父進程ID都是master的進程ID(1325)。

          [root@master ~]# ps -ef | grep nginx | grep -v grep | grep -v php-fpm
          root       1325      1  0 11:28 ?        00:00:00 nginx: master process /usr/local/nginx/sbin/nginx
          nginx      1332   1325  0 11:28 ?        00:00:00 nginx: worker process
          nginx      1334   1325  0 11:28 ?        00:00:00 nginx: worker process
          nginx      1335   1325  0 11:28 ?        00:00:00 nginx: worker process
          nginx      1336   1325  0 11:28 ?        00:00:00 nginx: worker process

          圖3-2 一個master進程與四個worker子進程

          我們可以通過 lsof -i:nginx端口號 來查看我們的master和worker進程。

          [root@master ~]# lsof -i:80
          COMMAND  PID  USER   FD   TYPE DEVICE SIZE/OFF NODE NAME
          nginx   1325  root    6u  IPv4  22282      0t0  TCP *:http (LISTEN)
          nginx   1332 nginx    6u  IPv4  22282      0t0  TCP *:http (LISTEN)
          nginx   1334 nginx    6u  IPv4  22282      0t0  TCP *:http (LISTEN)
          nginx   1335 nginx    6u  IPv4  22282      0t0  TCP *:http (LISTEN)
          nginx   1336 nginx    6u  IPv4  22282      0t0  TCP *:http (LISTEN)

          信號量管理

          Linux的信號量管理機制

          信號是進程間通信方式之一,典型用法是:終端用戶輸入中斷命令,通過信號機制停止一個程序的運行。

          我們可以通過給進程發(fā)送信號來管理我們的進程,kill -l命令可以查看linux支持的信號量

          linux信號量

          一共有64號信號量,主要需要弄清如下幾個:

          kill -1 $PID:(SIGHUP)重新加載進程,對于與終端脫離關(guān)系的守護進程,這個信號用于通知它重新讀取配置文件;

          kill -2 $PID:(SIGINT)中斷(通Ctrl+C);

          kill -3 $PID:(SIGQUIT)從鍵盤輸入的退出(ctrl-\);

          kill -9 $PID:(SIGKILL)立即殺死進程,無論當前程序處于什么狀態(tài);

          kill -10 $PID:(SIGUSR1)$USR1和$USR2都是留給用戶自定義的信號量;

          kill -12 $PID:($IGUSR2)

          kill -15 $PID:(SIGTERM)正常停止一個進程;

          kill -17 $PID:(SIGCHLD)父子進程通信的信號量,父進程可以fork()出很多子進程,子進程掛掉會給父進程發(fā)送信號;

          kill 可將指定的信息送至程序。預設(shè)的信息為 SIGTERM(15),可將指定程序終止。若仍無法終止該程序,可使用 SIGKILL(9) 信息嘗試強制刪除程序。程序或工作的編號可利用 ps 指令或 jobs 指令查看。

          kill -l # 查看所有能夠支持的信號
          kill PID
          # 殺死一個進程
          kill 1024
          # 殺死多個進程 進程號之間用空格隔開
          kill 1024 2048
          # kill -9 表示立即強制結(jié)束進程
          kill -9 1024

          注:Ctrl+C:停止終端中正在運行的進程,Ctrl+C可以比較有好地中止終端中正在運行的程序(進程)

          利用信號量管理Nginx進程

          管理Nginx進程可以這些方式:master進程、worker進程、命令行

          使用信號量管理master和worker(不推薦使用發(fā)送信號量的方式來管理worker進程,worker進程應(yīng)該交給master進程來管理和維護)。

          Master進程

          • 監(jiān)控worker進程

            • CHLD

          • 管理worker進程

          • 接收信號

            • TERM、INT

            • QUIT

            • HUP

            • USR1

            • USR2

            • WINCH

          示例:

          通過kill命令殺死m(xù)aster進程

          kill -s SIGTERM 1325

          通過kill命令讓Nginx重新讀取文件,這樣會關(guān)閉就得worker進程,生成新的worker進程,master進程(ID)依舊保持不變

          kill -s SIGHUP 1325

          Worker進程

          • 接收信號

            • TERM、INT

            • QUIT

            • USR1

            • WINCH

          • 雖然可以,但是不推薦使用信號量方式直接管理worker進程,worker進程應(yīng)該交給master進程來管理和維護

          示例:

          使用kill命令殺死一個worker進程,這樣會殺死一個worker進程,linux會殺掉的worker進程的父進程(master進程)發(fā)送SIGCHLD信號量,所以master進程監(jiān)測到我們某一個子進程可能出了問題,會啟動一個新的worker進程,維護worker進程的數(shù)量。

          kill -s SIGTERM 1332

          命令行

          • reload:HUP

          • reopen:USR2

          • stop:TERM

          • quit:QUIT

          可以使用nginx -h查看幫助命令

          [itbsl@master ~]$ nginx -h
          nginx version: nginx/1.18.0
          Usage: nginx [-?hvVtTq] [-s signal] [-c filename] [-p prefix] [-g directives]

          Options:
            -?,-h         : this help
            -v            : show version and exit
            -V            : show version and configure options then exit
            -t            : test configuration and exit
            -T            : test configuration, dump it and exit
            -q            : suppress non-error messages during configuration testing
            -s signal     : send signal to a master process: stop, quit, reopen, reload
            -p prefix     : set prefix path (default: /usr/local/nginx-1.18.0/)
            -c filename   : set configuration file (default: conf/nginx.conf)
            -g directives : set global directives out of configuration file

          參數(shù)說明:

          • -?,-h:查看幫助

          • -v:查看Nginx版本

          • -V:查看Nginx版本和編譯選項

          • -t:檢查配置文件語法是否正確

          • -T:檢查配置文件語法是否正確,并打印

          • -q:在檢查配置文件時不顯示非錯誤消息

          • -s:給master進程發(fā)送信號,可以發(fā)送:stop、quit、reopen、reload

          • -c:指定配置文件

          • -g:設(shè)置配置文件之外的全局指令

          配置文件重載原理

          我們知道了可以通過給nginx的master進程發(fā)送SIGHUP信號,或者使用nginx -s reload命令來達到重新載入配置文件,從而使nginx平滑升級。那我們執(zhí)行這樣一個命令之后,對nginx本身來說背后發(fā)生了什么事情呢,它是如何保證新老請求如何平滑過渡的?

          reload重載配置文件的流程

          1. 向master進程發(fā)送HUP信號(reload命令)

          2. master進程檢查配置語法是否正確

          3. master進程打開監(jiān)聽端口(在修改配置文件的端口情況下,可能)

          4. master進程使用新的配置文件啟動新的worker子進程

          5. master進程向老的worker子進程發(fā)送QUIT信號

          6. 舊的worker進程關(guān)閉監(jiān)聽句柄,處理完當前連接后關(guān)閉進程

          如果用圖示來描述的話大概如下圖所示

          nginx -s reload

          圖示解析:

          1.左邊綠色的狀態(tài)是執(zhí)行nginx -s reload命令之前的狀態(tài),按照我個人主機的配置時一個master進程和4個worker子進程。

          2.為了模擬執(zhí)行nginx -s reload命令后原來的worker進程會處理完請求后再被殺掉,我模擬一個需要很久才能處理完任務(wù)并響應(yīng)的接口,是的,我在代碼里sleep 15秒,也就是說這個接口響應(yīng)需要15秒,時間弄長點方便我們來觀察中間態(tài),注意,在執(zhí)行reload命令前請求該接口

          <?php
              sleep(15);
              echo json_encode(['msg' => 'hello world']);die();

          3.我們已經(jīng)知道了master進程會把任務(wù)交給worker子進程處理,目前只有一個任務(wù),所以當前只需要一個worker進程需要處理任務(wù)。

          4.執(zhí)行reload命令,master進程會創(chuàng)建4個(與你配置有關(guān))新的worker進程(我上圖中的黃色worker進程),關(guān)閉掉舊的空閑worker進程(綠色worker進程),而正在處理請求的舊worker進程不會立即關(guān)閉,而是會等請求處理完畢就關(guān)閉。

          5.剩下的最后一個舊worker進程任務(wù)處理完畢也被關(guān)掉,最后剩下的都是使用新nginx.conf配置產(chǎn)生的新worker進程,可以看下面的這張圖,那個處于is shutting down的舊worker進程就是因為處理上面sleep 15秒的任務(wù)接口還沒處理完畢,所以依然能夠被看到。





          粉絲福利:Java從入門到入土學習路線圖

          ??????

          ??長按上方微信二維碼 2 秒


          感謝點贊支持下哈 

          瀏覽 50
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  麻豆 传媒 国产 富婆 | 黄色一级一级 | 大鸡巴日逼 | 四虎殴美| 成人性生活免费视频 |