Nginx+Php-fpm运行原理详解

48

一、代理与反向代理

现实生活中的例子

1、正向代理:访问google.com

image_1b08aqr1s1ef3me7c56p061sj31g.png-9.3kB

如上图,因为google被墙,我们需要vpn翻墙才能访问google.com。

vpn对于“我们”来说,是可以感知到的(我们连接vpn)
vpn对于"google服务器"来说,是不可感知的(google只知道有http请求过来)。

对于人来说可以感知到,但服务器感知不到的服务器,我们叫他正向代理服务器。

2、反向代理:通过反向代理实现负载均衡

image_1b08aq6gvlbqnfp13k21odl1pr713.png-18.7kB

如上图,我们访问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文件
image_1b08eroasu1dr0kk0m12pg1572d.png-84.6kB

2、添加对应的server

进入上面include的路径,添加一个server.
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文件,我们看到如下配置:
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模块

image_1b08gunh4cf81rt61cbrr9m1og73k.png-47.1kB

启动成功,我们查看php-fpm进程
image_1b08h6i849is2hkqs1uct136h4e.png-74.9kB

如上图,有一个master进程,3个worker进程。

2、在网站目录下建立文件

我们编辑文件如下图:
image_1b08hafbi8lvr161pf21tq91f8a4r.png-23.8kB

3、访问网站

image_1b08hctce1ucf1e081jk6151kpcn58.png-88.3kB

更多精彩,请关注公众号“聊聊代码”,让我们一起聊聊“左手代码右手诗”的事儿。
图片描述

你可能感兴趣的

14 条评论
huier · 4月17日

有误,fastcgi不是进程,而是协议。是php-fpm管理进程的。所以我理解不是fast-cgi监听127.0.0.1:9000地址,而是php-fpm监听

+1 回复

0

对的,fastcgi是协议,应该是nginx的fcgi模块请求php-fpm开启的fast-cgi对应的9000端口。在这里nginx是client,php-fpm是server

若疯 · 6月4日
韦轩 · 2016年10月30日

图是用什么软件画?

回复

优de良 · 2016年10月30日

应该就是markdown

回复

zilu 作者 · 2016年10月31日

mac下面的omigraff

回复

ahinap · 2016年11月02日

谢谢

回复

m1509001 · 2017年05月26日

thank you

回复

changeluo · 2017年08月09日

浅显易懂,赞

回复

胖哥初来乍到 · 2017年11月08日

讲的很好

回复

iYting · 3月15日

您好! 我想转载你的文章

回复

FernandoIsco · 4月12日

有误,并不是路由到www.example.com/index.php,应该是默认访问根目录下这个文件。
不会执行两个location指令,根据匹配规则有一定的顺序,/ 匹配规则是最后才匹配的。

回复

若疯 · 6月4日

nginx的fast-cgi模块是“ 请求 ”127.0.0.1:9000,不是“监听”, fpm才是监听

回复

0

准确的说是转发,不是请求

王成涛 · 7月11日
常规的检查 · 8月2日

源码安装了nginx+php,业余时间搞了好几天没解析成功php文件,于是放弃排查错误
直接尝试理解php-fpm原理,才找到这篇文章,很浅显,非常感谢。

回复

载入中...