PHP-FPM和Nginx的通信机制详解


Posted in PHP onFebruary 01, 2019

PHP-FPM 介绍

CGI 协议与 FastCGI 协议

每种动态语言( PHP,Python 等)的代码文件需要通过对应的解析器才能被服务器识别,而 CGI 协议就是用来使解释器与服务器可以互相通信。PHP 文件在服务器上的解析需要用到 PHP 解释器,再加上对应的 CGI 协议,从而使服务器可以解析到 PHP 文件。

由于 CGI 的机制是每处理一个请求需要 fork 一个 CGI 进程,请求结束再kill掉这个进程,在实际应用上比较浪费资源,于是就出现了CGI 的改良版本 FastCGI,FastCGI 在请求处理完后,不会 kill 掉进程,而是继续处理多个请求,这样就大大提高了效率。

PHP-FPM 是什么

PHP-FPM 即 PHP-FastCGI Process Manager, 它是 FastCGI 的实现,并提供了进程管理的功能。进程包含 master 进程和 worker 进程两种;master 进程只有一个,负责监听端口,接收来自服务器的请求,而 worker 进程则一般有多个(具体数量根据实际需要进行配置),每个进程内部都会嵌入一个 PHP 解释器,是代码真正执行的地方。

Nginx 与 php-fpm 通信机制

当我们访问一个网站(如 www.test.com)的时候,处理流程是这样的:

  www.test.com
        |
        |
      Nginx
        |
        |
路由到 www.test.com/index.php
        |
        |
加载 nginx 的 fast-cgi 模块
        |
        |
fast-cgi 监听 127.0.0.1:9000 地址
        |
        |
www.test.com/index.php 请求到达 127.0.0.1:9000
        |
        |
     等待处理...

Nginx 与 php-fpm 的结合

在 Linux 上,nginx 与 php-fpm 的通信有 tcp socket 和 unix socket 两种方式。

tcp socket 的优点是可以跨服务器,当 nginx 和 php-fpm 不在同一台机器上时,只能使用这种方式。

Unix socket 又叫 IPC(inter-process communication 进程间通信) socket,用于实现同一主机上的进程间通信,这种方式需要在 nginx配置文件中填写 php-fpm 的 socket 文件位置。

两种方式的数据传输过程如下图所示:

PHP-FPM和Nginx的通信机制详解

二者的不同:

由于 Unix socket 不需要经过网络协议栈,不需要打包拆包、计算校验和、维护序号和应答等,只是将应用层数据从一个进程拷贝到另一个进程。所以其效率比 tcp socket 的方式要高,可减少不必要的 tcp 开销。不过,unix socket 高并发时不稳定,连接数爆发时,会产生大量的长时缓存,在没有面向连接协议的支撑下,大数据包可能会直接出错不返回异常。而 tcp 这样的面向连接的协议,可以更好的保证通信的正确性和完整性。

Nginx 与 php-fpm 结合只需要在各自的配置文件中做设置即可:

1) Nginx 中的配置

以 tcp socket通信为例

server {
 listen  80; #监听 80 端口,接收http请求
 server_name www.test.com; #就是网站地址
 root /usr/local/etc/nginx/www/huxintong_admin; # 准备存放代码工程的路径
 #路由到网站根目录 www.test.com 时候的处理
 location / {
  index index.php; #跳转到 www.test.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; # tcp 方式,php-fpm 监听的 IP 地址和端口
  # fasrcgi_pass /usr/run/php-fpm.sock # unix socket 连接方式
 }

}

2) php-fpm 的配置

listen = 127.0.0.1:9000
# 或者下面这样
listen = /var/run/php-fpm.sock

注意,在使用 unix socket 方式连接时,由于 socket 文件本质上是一个文件,存在权限控制的问题,所以需要注意 nginx 进程的权限与 php-fpm 的权限问题,不然会提示无权限访问。(在各自的配置文件里设置用户)

通过以上配置即可完成 php-fpm 与 nginx 的通信。

在应用中的选择

如果是在同一台服务器上运行的 nginx 和 php-fpm,且并发量不高(不超过1000),选择unix socket,以提高 nginx 和 php-fpm 的通信效率。

如果是面临高并发业务,则考虑选择使用更可靠的 tcp socket,以负载均衡、内核优化等运维手段维持效率。

若并发较高但仍想用 unix socket 时,可通过以下方式提高 unix socket 的稳定性。

1)将sock文件放在 /dev/shm 目录下,此目录下将 sock 文件放在内存里面,内存的读写更快。

2)提高 backlog

backlog 默认位 128,1024 这个值最好换算成自己正常的 QPS,配置如下。

nginx.conf 文件中

server {
  listen 80 default backlog = 1024;
  }

php-fpm.conf 文件中

listen.backlog = 1024

3)增加 sock 文件和 php-fpm 实例

在 /dev/shm 新建一个 sock 文件,在 nginx 中通过 upstream 模块将请求负载均衡到两个 sock 文件,并且将两个 sock 文件分别对应到两套 php-fpm 实例上。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
PHP个人网站架设连环讲(三)
Oct 09 PHP
浅析linux下apache服务器的配置和管理
Aug 10 PHP
php指定函数参数默认值示例代码
Dec 04 PHP
采用header定义为文件然后readfile下载(隐藏下载地址)
Jan 31 PHP
php数组去重复数据示例
Feb 25 PHP
PHP使用mysql_fetch_row查询获得数据行列表的方法
Mar 18 PHP
php使用for语句输出三角形的方法
Jun 09 PHP
详谈PHP中的密码安全性Password Hashing
Feb 04 PHP
实例讲解PHP验证邮箱是否合格
Jan 28 PHP
php实现文章评论系统
Feb 18 PHP
thinkPHP5.1框架中Request类四种调用方式示例
Aug 03 PHP
PHP常用函数之获取汉字首字母功能示例
Oct 21 PHP
Laravel框架自定义验证过程实例分析
Feb 01 #PHP
thinkPHP框架RBAC实现原理分析
Feb 01 #PHP
PHP PDOStatement::fetchColumn讲解
Jan 31 #PHP
PHP PDOStatement::fetchAll讲解
Jan 31 #PHP
PHP PDOStatement::fetch讲解
Jan 31 #PHP
PHP PDOStatement::execute讲解
Jan 31 #PHP
PHP PDOStatement::errorInfo讲解
Jan 31 #PHP
You might like
星际争霸中的对战模式介绍
2020/03/04 星际争霸
php通过字符串调用函数示例
2014/03/02 PHP
三个思路解决laravel上传文件报错:413 Request Entity Too Large问题
2017/11/13 PHP
PHP xpath提取网页数据内容代码解析
2020/07/16 PHP
php解析非标准json、非规范json的方式实例
2020/12/10 PHP
模仿JQuery sortable效果 代码有错但值得看看
2009/11/05 Javascript
情人节专属 纯js脚本1k大小的3D玫瑰效果
2012/02/11 Javascript
js Select下拉列表框进行多选、移除、交换内容的具体实现方法
2013/08/13 Javascript
jQuery制作仿腾讯web qq用户体验桌面
2013/08/20 Javascript
三种动态加载js的jquery实例代码另附去除js方法
2014/04/30 Javascript
javascript实现节点(div)名称编辑
2014/12/17 Javascript
微信小程序 简单教程实例详解
2017/01/13 Javascript
关于webpack2和模块打包的新手指南(小结)
2017/08/07 Javascript
dropload.js插件下拉刷新和上拉加载使用详解
2017/10/20 Javascript
vue中使用ueditor富文本编辑器
2018/02/08 Javascript
详解在Node.js中发起HTTP请求的5种方法
2019/01/10 Javascript
微信小程序实现工作时间段选择
2019/02/15 Javascript
Bootstarp在pycharm中的安装及简单的使用方法
2019/04/19 Javascript
vue-router 前端路由之路由传值的方式详解
2019/04/30 Javascript
javascript+css实现进度条效果
2020/03/25 Javascript
Python实现获取网站PR及百度权重
2015/01/21 Python
编写Python CGI脚本的教程
2015/06/29 Python
Python FTP两个文件夹间的同步实例代码
2018/05/25 Python
python实现微信小程序自动回复
2018/09/10 Python
Python中调用其他程序的方式详解
2019/08/06 Python
基于Django框架的权限组件rbac实例讲解
2019/08/31 Python
Python SELENIUM上传文件或图片实现过程
2019/10/28 Python
利用Python制作动态排名图的实现代码
2020/04/09 Python
浅析NumPy 切片和索引
2020/09/02 Python
基于CSS3特效之动画:animation的应用
2013/05/09 HTML / CSS
Blue Nile蓝色尼罗河香港官网:世界最大在线钻石珠宝销售商
2020/05/07 全球购物
激情洋溢的毕业生就业求职信
2014/03/15 职场文书
教师暑期培训感言
2014/08/15 职场文书
小学毕业典礼演讲稿
2014/09/09 职场文书
html form表单基础入门案例讲解
2021/07/21 HTML / CSS
本地搭建minio文件服务器(使用bat脚本启动)的方法
2022/07/15 Servers