Nginx+Php-fpm 執行原理詳解

子路發表於2017-03-29

一、代理與反向代理

現實生活中的例子

1、正向代理:訪問google.com

Nginx+Php-fpm 執行原理詳解

如上圖,因為google被牆,我們需要vpn翻牆才能訪問google.com。

vpn對於“我們”來說,是可以感知到的(我們連線vpn)
vpn對於"google伺服器"來說,是不可感知的(google只知道有http請求過來)。
對於人來說可以感知到,但伺服器感知不到的伺服器,我們叫他正向代理伺服器。

2、反向代理:通過反向代理實現負載均衡

Nginx+Php-fpm 執行原理詳解

如上圖,我們訪問baidu.com的時候,baidu有一個代理伺服器,通過這個代理伺服器,可以做負載均衡,路由到不同的server。

此代理伺服器,對於“我們”來說是不可感知的(我們只能感知到訪問的是百度的伺服器,不知道中間還有代理伺服器來做負載均衡)。
此代理伺服器,對於"server1 server2 server3"是可感知的(代理伺服器負載均衡路由到不同的server)
對於人來說不可感知,但對於伺服器來說是可以感知的,我們叫他反向代理伺服器

總結

說白了:“正向”、“反向”是相對於人的感知來說的。
人能感受到的代理就是正向代理,人感受不到的代理就是反向代理。

二、初識Nginx與Php-fpm

Nginx是什麼

Nginx ("engine x") 是一個高效能的HTTP和反向代理伺服器,也是一個IMAP/POP3/SMTP伺服器。

Php-fpm是什麼

1、cgi、fast-cgi協議
cgi的歷史

早期的webserver只處理html等靜態檔案,但是隨著技術的發展,出現了像php等動態語言。
webserver處理不了了,怎麼辦呢?那就交給php直譯器來處理吧!
交給php直譯器處理很好,但是,php直譯器如何與webserver進行通訊呢?
為了解決不同的語言直譯器(如php、python直譯器)與webserver的通訊,於是出現了cgi協議。只要你按照cgi協議去編寫程式,就能實現語言直譯器與webwerver的通訊。如php-cgi程式。

fast-cgi的改進

有了cgi協議,解決了php直譯器與webserver通訊的問題,webserver終於可以處理動態語言了。
但是,webserver每收到一個請求,都會去fork一個cgi程式,請求結束再kill掉這個程式。這樣有10000個請求,就需要fork、kill php-cgi程式10000次。

有沒有發現很浪費資源?

於是,出現了cgi的改良版本,fast-cgi。fast-cgi每次處理完請求後,不會kill掉這個程式,而是保留這個程式,使這個程式可以一次處理多個請求。這樣每次就不用重新fork一個程式了,大大提高了效率。

2、php-fpm是什麼

php-fpm即php-Fastcgi Process Manager.
php-fpm是 FastCGI 的實現,並提供了程式管理的功能。
程式包含 master 程式和 worker 程式兩種程式。
master 程式只有一個,負責監聽埠,接收來自 Web Server 的請求,而 worker 程式則一般有多個(具體數量根據實際需要配置),每個程式內部都嵌入了一個 PHP 直譯器,是 PHP 程式碼真正執行的地方。

三、Nginx如何與Php-fpm結合

上面我們說了,Nginx不只有處理http請求的功能,還能做反向代理。
Nginx通過反向代理功能將動態請求轉向後端Php-fpm。

下面我們來配置一個全新的Nginx+Php-fpm

1、配置nginx.conf檔案

進入nginx目錄下,編輯 nginx.conf檔案。
如圖,在nginx.conf最後一行,新增include檔案

Nginx+Php-fpm 執行原理詳解
image_1b08eroasu1dr0kk0m12pg1572d.png-84.6kB

2、新增對應的server

進入上面include的路徑,新增一個server.

Nginx+Php-fpm 執行原理詳解
image_1b08f7sqm1ub71es9jrr1v3h1emp2q.png-119kB

下面我們解釋一下配置項的含義:

server {
    listen       80; #監聽80埠,接收http請求
    server_name  www.example.com; #就是網站地址
    root /usr/local/etc/nginx/www/huxintong_admin; # 準備存放程式碼工程的路徑
    #路由到網站根目錄www.example.com時候的處理
    location / {
        index index.php; #跳轉到www.example.com/index.php
        autoindex on;
    }   

    #當請求網站下php檔案的時候,反向代理到php-fpm
    location ~ \.php$ {
        include /usr/local/etc/nginx/fastcgi.conf; #載入nginx的fastcgi模組
        fastcgi_intercept_errors on;
        fastcgi_pass   127.0.0.1:9000; #nginx fastcgi程式監聽的IP地址和埠
    }

}複製程式碼

總而言之:當我們訪問www.example.com的時候,處理流程是這樣的:

  www.example.com
        |
        |
      Nginx
        |
        |
路由到www.example.com/index.php
        |
        |
載入nginx的fast-cgi模組
        |
        |
fast-cgi監聽127.0.0.1:9000地址
        |
        |
www.example.com/index.php請求到達127.0.0.1:9000
        |
        |
     等待處理...複製程式碼

下面我們啟用php的php-fpm來處理這個請求

開啟php-fpm.conf檔案,我們看到如下配置:

Nginx+Php-fpm 執行原理詳解
image_1b08gcs3g1msg9mf1ie7rks1b7o37.png-96kB

即:php-fpm模組監聽127.0.0.1:9000埠,等待請求到來去處理。

四、總結

nginx與php-fpm的結合,完整的流程是這樣的。

 www.example.com
        |
        |
      Nginx
        |
        |
路由到www.example.com/index.php
        |
        |
載入nginx的fast-cgi模組
        |
        |
fast-cgi監聽127.0.0.1:9000地址
        |
        |
www.example.com/index.php請求到達127.0.0.1:9000
        |
        |
php-fpm 監聽127.0.0.1:9000
        |
        |
php-fpm 接收到請求,啟用worker程式處理請求
        |
        |
php-fpm 處理完請求,返回給nginx
        |
        |
nginx將結果通過http返回給瀏覽器複製程式碼

五、效果展示

1、啟動nginx與php-fpm模組

Nginx+Php-fpm 執行原理詳解

啟動成功,我們檢視php-fpm程式

Nginx+Php-fpm 執行原理詳解

如上圖,有一個master程式,3個worker程式。

2、在網站目錄下建立檔案

我們編輯檔案如下圖:

Nginx+Php-fpm 執行原理詳解

3、訪問網站

Nginx+Php-fpm 執行原理詳解

更多精彩,請關注公眾號“聊聊程式碼”,讓我們一起聊聊“左手程式碼右手詩”的事兒。

Nginx+Php-fpm 執行原理詳解

相關文章