LNMP架构介绍
和LAMP架构类似,唯一不同的就是提供web服务的是Nginx,而不是Apache。
并且php是作为一个独立服务存在的,这个服务叫做php-fpm
Nginx直接处理静态请求,动态请求会转发给php-fpm
原理是:lnmp里的php会启动一个服务,nginx把用户请求的php交给php-fpm处理,比如登录网站php要和mysql做交互,查用户的账号和密码,处理完之后,php-fpm会告诉nginx,nginx返回给用户一个结果。Nginx对于一些静态请求就自己处理了,不需要交给php处理。
mysql安装
由于之前已经安装过mysql,这里可以删除重装一遍,步骤和以前相同。
删掉一下文件:
# rm -rf /usr/local/mysql
删除启动脚本:
# rm -rf /etc/init.d/mysqld
删除/data/mysql:
# rm -rf /data/mysql/*
MySQL的配置文件(/etc/my.cnf)可以不用删
MySQL的几个常用安装包:rpm、源码、二进制免编译
追求性能可以自己编译,否则可以使用二进制免编译
下载地址可到r.aminglinux.com下查找
下载MySQL到指定目录下(/usr/local/src).
[root@linux7-128 ~]# cd /usr/local/src/
[root@linux7-128 src]# wget http://mirrors.163.com/mysql/Downloads/MySQL-5.6/mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz
--2018-09-07 22:11:05-- http://mirrors.163.com/mysql/Downloads/MySQL-5.6/mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz
正在解析主机 mirrors.163.com (mirrors.163.com)... 59.111.0.251
正在连接 mirrors.163.com (mirrors.163.com)|59.111.0.251|:80... 已连接。
已发出 HTTP 请求,正在等待回应... 200 OK
长度:328882304 (314M) [application/octet-stream]
正在保存至: “mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz”
100%[===============================================================================>] 328,882,304 891KB/s 用时 4m 52s
2018-09-07 22:15:57 (1.08 MB/s) - 已保存 “mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz” [328882304/328882304])
解压
[root@linux7-128 src]# tar zxvf mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz
目录改名为mysql
[root@linux7-128 src]# mv mysql-5.6.39-linux-glibc2.12-x86_64 /usr/local/mysql
进入到mysql目录下
[root@linux7-128 src]# cd /usr/local/mysql/
创建一个新用户mysql,并且新建一个data目录用于存放数据,如果data已存在可以省略
[root@linux7-128 mysql]# useradd mysql
[root@linux7-128 mysql]# mkdir /data/
mkdir: 无法创建目录"/data/": 文件已存在
[root@linux7-128 mysql]# ls
bin COPYING data docs include lib man mysql-test README scripts share sql-bench support-files
初始化,生成/data/mysql目录,如果mysql想要启动必须要有一个自带的库mysql
[root@linux7-128 mysql]# ./scripts/mysql_install_db --user=mysql --datadir=/data/mysql
没有报错,初始化完成。
[root@linux7-128 mysql]# echo $?
0 --echo $? 是检测上一条命令是否执行成功
此步骤可以省略
拷贝配置文件到/etc/目录下并改名为my.cnf,如果不放到etc目录下那么启动时要做一个指定
[root@linux7-128 mysql]# cp support-files/my-default.cnf /etc/my.cnf
其实/etc下默认已经有了一个my.cnf
[root@linux7-128 mysql]# ls /etc/my.cnf
/etc/my.cnf
查看是由哪个包生成的my.cnf文件
[root@linux7-128 mysql]# rpm -qf /etc/my.cnf
mariadb-libs-5.5.56-2.el7.x86_64
--centos7可能默认安装了此包
自带的/etc/my.cnf也可以使用,前提是要更改下文件设置
[mysqld]
datadir=/data/mysql
socket=/tmp/mysql.sock
# Disabling symbolic-links is recommended to prevent assorted security risks
symbolic-links=0
# Settings user and group are ignored when systemd is used.
# If you need to run mysqld under a different user or group,
# customize your systemd unit file for mariadb according to the
# instructions in http://fedoraproject.org/wiki/Systemd
[mysqld_safe]
#log-error=/var/log/mariadb/mariadb.log
#pid-file=/var/run/mariadb/mariadb.pid
#
# include all files from the config directory
#
#!includedir /etc/my.cnf.d
- 将
datadir
位置修改为/data/mysql
- 将
socke
位置修改为/tmp/mysql.sock
- 注释掉
log-error=/var/log/mariadb/mariadb.log
- 注释掉
pid-file=/var/run/mariadb/mariadb.pid
- 注释掉
!includedir /etc/my.cnf.d
复制启动脚本到/etc/init.d
下并命名为mysqld
[root@linux7-128 mysql]# cp support-files/mysql.server /etc/init.d/mysqld
修改配置文件/etc/init.d/mysqld
- 指定程序目录
basedir=/usr/local/mysql
- 指定数据目录
datadir=/data/mysql
查看文件权限,如果不是755修改为755
[root@linux7-128 mysql]# ls -l /etc/init.d/mysqld
-rwxr-xr-x 1 root root 10592 9月 7 23:16 /etc/init.d/mysqld
设置脚本文件为开机启动
[root@linux7-128 mysql]# chkconfig --add mysqld
[root@linux7-128 mysql]# chkconfig --list
注:该输出结果只显示 SysV 服务,并不包含
原生 systemd 服务。SysV 配置数据
可能被原生 systemd 配置覆盖。
要列出 systemd 服务,请执行 'systemctl list-unit-files'。
查看在具体 target 启用的服务请执行
'systemctl list-dependencies [target]'。
mysqld 0:关 1:关 2:开 3:开 4:开 5:开 6:关
netconsole 0:关 1:关 2:关 3:关 4:关 5:关 6:关
network 0:关 1:关 2:开 3:关 4:开 5:开 6:关
也可以手动启动
[root@linux7-128 mysql]# service mysqld start
Starting MySQL.Logging to '/data/mysql/linux7-128.err'.
. SUCCESS!
或者
# /etc/init.d/mysqld start
Starting MySQL.Logging to '/data/mysql/linux7-128.err'.
... SUCCESS!
**注意:–defaults-file=/etc/my.cnf 指定配置文件路径 **
PHP安装
和LAMP安装PHP方法有差别,这里需要开启php-fpm服务,之前php是作为Apache的一个模块,需要指定Apache的路径,用apxs工具配置模块怎么加载。这里就不需要了,因为用不到Apache。Php是作为独立的服务。
首先进入到/usr/local/src里的PHP目录,清除掉之前的编译进入的初始状态
[root@iz2zef1im6qv29viqhtk3qz ~]# cd /usr/local/src/
[root@iz2zef1im6qv29viqhtk3qz src]# ls
apr-1.6.3 httpd-2.4.34 php-5.6.32.tar.bz2
apr-1.6.3.tar.gz httpd-2.4.34.tar.gz php-7.1.6
apr-util-1.6.1 mariadb-10.2.6-linux-glibc_214-x86_64.tar.gz php-7.1.6.tar.bz2
apr-util-1.6.1.tar.bz2 mysql-5.6.39-linux-glibc2.12-x86_64.tar.gz
apr-util-1.6.1.tar.gz php-5.6.32
[root@iz2zef1im6qv29viqhtk3qz src]# cd php-5.6.32
[root@iz2zef1im6qv29viqhtk3qz php-5.6.32]# make clean
find . -name \*.gcno -o -name \*.gcda | xargs rm -f
find . -name \*.lo -o -name \*.o | xargs rm -f
find . -name \*.la -o -name \*.a | xargs rm -f
find . -name \*.so | xargs rm -f
find . -name .libs -a -type d|xargs rm -rf
rm -f libphp5.la sapi/cli/php sapi/cgi/php-cgi libphp5.la modules/* libs/*
然后对新文件夹取名为php-fpm,编译安装
./configure --prefix=/usr/local/php-fpm --with-config-file-path=/usr/local/php-fpm/etc --enable-fpm --with-fpm-user=php-fpm --with-fpm-group=php-fpm --with-mysql=/usr/local/mysql --with-mysqli=/usr/local/mysql/bin/mysql_config --with-pdo-mysql=/usr/local/mysql --with-mysql-sock=/tmp/mysql.sock --with-libxml-dir --with-gd --with-jpeg-dir --with-png-dir --with-freetype-dir --with-iconv-dir --with-zlib-dir --with-mcrypt --enable-soap --enable-gd-native-ttf --enable-ftp --enable-mbstring --enable-exif --with-pear --with-curl --with-openssl
- –prefix 指定路径
- –with-config-file-path 指定配置文件所在路径
- –enable-fpm 要加上这个,如果不加就不会有php-fpm执行文件生成,更不能启动php-fpm服务。
- –with-fpm-user=php-fpm --with-fpm-group 指定用户和组
- –with-mysql 指定mysql路径,后面的mysqli,pdo-mysql,mysql-sock也一样
有报错信息提示:
checking for cURL in default path... not found
configure: error: Please reinstall the libcurl distribution -
easy.h should be in <curl-dir>/include/curl/
解决办法:安装cURL包
之前有提到过不知道安装哪个包可以使用yum list命令,我们一般安装lib.devel包
# yum list |grep curl
# yum install -y libcurl-devel
然后继续编辑,编译完成。
# echo $?
0
接下来开始使用make和make install安装
# make
# make install
安装完成之后可以看到比原先的PHP文件多了两个文件夹
[root@iz2zef1im6qv29viqhtk3qz php-5.6.32]# ls /usr/local/php-fpm/
bin etc include lib php sbin var
[root@iz2zef1im6qv29viqhtk3qz php-5.6.32]# ls /usr/local/php/
bin etc include lib php
在sbin目录下有个php-fpm文件,他就是启动php-fpm的文件
# ls /usr/local/php-fpm/sbin/
php-fpm
ls /usr/local/php-fpm/var/log/
//存放日志
ls /usr/local/php-fpm/var/run/
//存放pid
可以使用-i 和-m查看一些信息,比如模块
# /usr/local/php-fpm/sbin/php-fpm -m
[PHP Modules]
cgi-fcgi
Core
ctype
curl
date
dom
ereg
省略。。。
或者
# /usr/local/php-fpm/bin/php -m
[PHP Modules]
Core
ctype
curl
date
dom
ereg
省略。。。
两个是一样的
创建php-fpm用户useradd php-fpm
php-fpm测试自己的配置文件语法是否正确
# /usr/local/php-fpm/sbin/php-fpm -t
[16-Sep-2018 23:14:10] ERROR: failed to open configuration file '/usr/local/php-fpm/etc/php-fpm.conf': No such file or directory (2)
[16-Sep-2018 23:14:10] ERROR: failed to load configuration file '/usr/local/php-fpm/etc/php-fpm.conf'
[16-Sep-2018 23:14:10] ERROR: FPM initialization failed
这里可以看到,配置文件不存在。要想启动php-fpm服务,要先给他配置配置文件。
在/usr/local/php-fpm/etc/下是没有php.ini文件的
[root@iz2zef1im6qv29viqhtk3qz ~]# cd /usr/local/php-fpm/etc/
[root@iz2zef1im6qv29viqhtk3qz etc]# ls
pear.conf php-fpm.conf.default
需要将PHP包下面的配置文件拷贝一份
[root@iz2zef1im6qv29viqhtk3qz etc]# cd /usr/local/src/php-5.6.32
[root@iz2zef1im6qv29viqhtk3qz php-5.6.32]# cp php.ini-production /usr/local/php-fpm/etc/php.ini
除了php.ini之外。还在在etc目录下创建一个php-fpm.conf文件,当然也可以把php-fpm.conf.default改名直接使用,编辑php-fpm.conf
# vim php-fpm.conf
所需代码地址:https://coding.net/u/aminglinux/p/aminglinux-book/git
第十五章里,拷贝后还需要加一行listen以后会用到,这里先注释掉。
[global]
pid = /usr/local/php-fpm/var/run/php-fpm.pid
error_log = /usr/local/php-fpm/var/log/php-fpm.log
[www] //一个模块的名字
listen = /tmp/php-fcgi.sock //监听的地址
#listen = 127.0.0.1:9000 //内部监听,一般PHP和Nginx在一台机器上
listen.mode = 666 //监听的是sock的这行才会生效,用来定义sock文件的权限是666
user = php-fpm
group = php-fpm
pm = dynamic
pm.max_children = 50
pm.start_servers = 20
pm.min_spare_servers = 5
pm.max_spare_servers = 35
pm.max_requests = 500
rlimit_files = 1024
保存退出
进入源码包目录拷贝启动脚本
[root@iz2zef1im6qv29viqhtk3qz etc]# cd /usr/local/src/php-5.6.32
[root@iz2zef1im6qv29viqhtk3qz php-5.6.32]# cp sapi/fpm/init.d.php-fpm /etc/init.d/php-fpm
更改权限并且加到服务列表下
chmod 755 /etc/init.d/php-fpm
chkconfig --add php-fpm //加入开机启动列表
chkconfig php-fpm on //开机启动
启动服务
# service php-fpm start
Starting php-fpm done
# ps aux |grep php-fpm
root 17656 0.0 0.4 123600 4944 ? Ss 21:50 0:00 php-fpm: master process (/usr/local/php-fpm/etc/php-fpm.conf)
php-fpm 17657 0.0 0.4 123600 4704 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17658 0.0 0.4 123600 4704 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17659 0.0 0.4 123600 4704 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17660 0.0 0.4 123600 4704 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17661 0.0 0.4 123600 4708 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17662 0.0 0.4 123600 4708 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17663 0.0 0.4 123600 4708 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17664 0.0 0.4 123600 4708 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17665 0.0 0.4 123600 4708 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17666 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17667 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17668 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17669 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17670 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17671 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17672 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17673 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17674 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17675 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
php-fpm 17676 0.0 0.4 123600 4712 ? S 21:50 0:00 php-fpm: pool www
root 17678 0.0 0.0 112660 968 pts/0 R+ 21:50 0:00 grep --color=auto php-fpm
这里的www就是前面我们配置文件里面定义的模块名。
Nginx介绍
- Nginx官网 nginx.org 下载带有stable的 http://nginx.org/
- Nginx应用场景:web服务、反向代理、负载均衡
- Nginx著名分支,淘宝基于Nginx开发的Tengine,使用上和Nginx一致,服务名,配置文件名都一样,和Nginx的最大区别在于Tenging增加了一些定制化模块,在安全限速方面表现突出,另外它支持对js,css合并
- Nginx核心+lua相关的组件和模块组成了一个支持lua的高性能web容器openresty,参考:http://jinnianshilongnian.iteye.com/blog/2280928
基本特性
nginx做为HTTP服务器,有以下几项基本特性:
- 处理静态文件,索引文件以及自动索引;打开文件描述符缓冲.
- 无缓存的反向代理加速,简单的负载均衡和容错.
- FastCGI,简单的负载均衡和容错.
- 模块化的结构。包括gzipping, byte ranges, chunked responses,以及 SSI-filter等filter。如果由FastCGI或其它代理服务器处理单页中存在的多个SSI,则这项处理可以并行运行,而不需要相互等待。
- 支持SSL 和 TLSSNI.
Nginx专为性能优化而开发,性能是其最重要的考量,实现上非常注重效率 。它支持内核Poll模型,能经受高负载的考验,有报告表明能支持高达 50,000个并发连接数。
Nginx具有很高的稳定性。其它HTTP服务器,当遇到访问的峰值,或者有人恶意发起慢速连接时,也很可能会导致服务器物理内存耗尽频繁交换,失去响应,只能重启服务器。例如当前apache一旦上到200个以上进程,web响应速度就明显非常缓慢了。而Nginx采取了分阶段资源分配技术,使得它的CPU与内存占用率非常低。nginx官方表示保持10,000个没有活动的连接,它只占2.5M内存,所以类似DOS这样的攻击对nginx来说基本上是毫无用处的。就稳定性而言,nginx比lighthttpd更胜一筹。
Nginx支持热部署。它的启动特别容易, 并且几乎可以做到7*24不间断运行,即使运行数个月也不需要重新启动。你还能够在不间断服务的情况下,对软件版本进行进行升级。
Nginx采用master-slave模型,能够充分利用SMP的优势,且能够减少工作进程在磁盘I/O的阻塞延迟。当采用select()/poll()调用时,还可以限制每个进程的连接数。
Nginx代码质量非常高,代码很规范,手法成熟, 模块扩展也很容易。特别值得一提的是强大的Upstream与Filter链。Upstream为诸如reverse proxy,与其他服务器通信模块的编写奠定了很好的基础。而Filter链最酷的部分就是各个filter不必等待前一个filter执行完毕。它可以把前一个filter的输出做为当前filter的输入,这有点像Unix的管线。这意味着,一个模块可以开始压缩从后端服务器发送过来的请求,且可以在模块接收完后端服务器的整个请求之前把压缩流转向客户端。
Nginx采用了一些os提供的最新特性如对sendfile (Linux2.2+),accept-filter (FreeBSD4.1+),TCP_DEFER_ACCEPT (Linux 2.4+)的支持,从而大大提高了性能。
当然,nginx还很年轻,多多少少存在一些问题,比如:Nginx是俄罗斯人创建,虽然前几年文档比较少,但是目前文档方面比较全面,英文资料居多,中文的资料也比较多,而且有专门的书籍和资料可供查找。
Nginx安装
下载源码包,解压
# cd /usr/local/src/
# wget http://nginx.org/download/nginx-1.8.0.tar.gz
# tar -zxvf nginx-1.8.0.tar.gz
编译安装
编译时根据自己需求配置
# ./configure --prefix=/usr/local/nginx
# make
# make install
安装完之后进入目录查看
# cd /usr/local/nginx/
# ls
conf html logs sbin
- conf : 配置文件目录
- html : 网页样例
- logs: 日志
- sbin: 进程,核心文件
配置文件在sbin目录下
# ls /usr/local/nginx/sbin/nginx
/usr/local/nginx/sbin/nginx
可以使用-t检查配置文件是否正确
# /usr/local/nginx/sbin/nginx -t
nginx: the configuration file /usr/local/nginx/conf/nginx.conf syntax is ok
nginx: configuration file /usr/local/nginx/conf/nginx.conf test is successful
配置启动脚本
/etc/init.d/nginx
vim /etc/init.d/nginx
内容可以拷贝如下内容:
#!/bin/bash
# chkconfig: - 30 21
# description: http service.
# Source Function Library
. /etc/init.d/functions
# Nginx Settings
NGINX_SBIN="/usr/local/nginx/sbin/nginx"
NGINX_CONF="/usr/local/nginx/conf/nginx.conf"
NGINX_PID="/usr/local/nginx/logs/nginx.pid"
RETVAL=0
prog="Nginx"
start()
{
echo -n $"Starting $prog: "
mkdir -p /dev/shm/nginx_temp
daemon $NGINX_SBIN -c $NGINX_CONF
RETVAL=$?
echo
return $RETVAL
}
stop()
{
echo -n $"Stopping $prog: "
killproc -p $NGINX_PID $NGINX_SBIN -TERM
rm -rf /dev/shm/nginx_temp
RETVAL=$?
echo
return $RETVAL
}
reload()
{
echo -n $"Reloading $prog: "
killproc -p $NGINX_PID $NGINX_SBIN -HUP
RETVAL=$?
echo
return $RETVAL
}
restart()
{
stop
start
}
configtest()
{
$NGINX_SBIN -c $NGINX_CONF -t
return 0
}
case "$1" in
start)
start
;;
stop)
stop
;;
reload)
reload
;;
restart)
restart
;;
configtest)
configtest
;;
*)
echo $"Usage: $0 {start|stop|reload|restart|configtest}"
RETVAL=1
esac
exit $RETVAL
更改权限,并添加系统服务
# chmod 755 /etc/init.d/nginx
# chkconfig --add nginx
# chkconfig nginx on
配置Nginx配置文件
Nginx的conf目录下有nginx.conf这个文件,我们可以将它拷贝一个重新自己创建一个
# cd /usr/local/nginx/conf
# mv nginx.conf nginx.conf.1
# vim nginx.conf
内容如下:
user nobody nobody; //访问一个图片,是以哪个用户去读这个图片用户和用户组
worker_processes 2; //定义子进程有几个
error_log /usr/local/nginx/logs/nginx_error.log crit; //debug或者crit这里表示错误级别
pid /usr/local/nginx/logs/nginx.pid;
worker_rlimit_nofile 51200; //nginx最多打开多少个文件
events
{
use epoll; //使用的模式
worker_connections 6000; //进程最大连接多少
}
http
{
include mime.types;
default_type application/octet-stream;
server_names_hash_bucket_size 3526;
server_names_hash_max_size 4096;
log_format combined_realip '$remote_addr $http_x_forwarded_for [$time_local]'
' $host "$request_uri" $status'
' "$http_referer" "$http_user_agent"';
sendfile on;
tcp_nopush on;
keepalive_timeout 30;
client_header_timeout 3m;
client_body_timeout 3m;
send_timeout 3m;
connection_pool_size 256;
client_header_buffer_size 1k;
large_client_header_buffers 8 4k;
request_pool_size 4k;
output_buffers 4 32k;
postpone_output 1460;
client_max_body_size 10m;
client_body_buffer_size 256k;
client_body_temp_path /usr/local/nginx/client_body_temp;
proxy_temp_path /usr/local/nginx/proxy_temp;
fastcgi_temp_path /usr/local/nginx/fastcgi_temp;
fastcgi_intercept_errors on;
tcp_nodelay on;
gzip on;
gzip_min_length 1k;
gzip_buffers 4 8k;
gzip_comp_level 5;
gzip_http_version 1.1;
gzip_types text/plain application/x-javascript text/css text/htm
application/xml;
server
{
listen 80;
server_name localhost; //server_name域名
index index.html index.htm index.php; //index定义首页格式
root /usr/local/nginx/html; //root 网站根目录
location ~ \.php$ //解析php部分
{
include fastcgi_params;
fastcgi_pass unix:/tmp/php-fcgi.sock; //nginx调用php-fpm,指定php监听的ip或者sock
#fastcgi_pass 127.0.0.1:9000; //和前面定义的php-fpm配置文件里的一样,指定监听的地址,此处先注释掉,可不用。
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME /usr/local/nginx/html$fastcgi_script_name;
}
}
}
这里的一个sever相当于前面http里的一台虚拟主机
检查配置文件是否正确
# /usr/local/nginx/sbin/nginx -t
nginx: the configuration file /usr/local/nginx/conf/nginx.conf syntax is ok
nginx: configuration file /usr/local/nginx/conf/nginx.conf test is successful
启动
/etc/init.d/nginx start
插曲:
如果之前将80端口配置给了其他服务,需要关闭服务并且杀死占用80端口的才可以正常启动Nginx。
参考博客:https://blog.csdn.net/ljfphp/article/details/79102840
nginx配置详解:
http://www.ha97.com/5194.html
https://my.oschina.net/duxuefeng/blog/34880
拓展
Nginx为什么比Apache Httpd高效:原理篇 http://www.toxingwang.com/linux-unix/linux-basic/1712.html
https://coding.net/u/aminglinux/p/nginx/git/blob/master/4z/IO.md?public=true
https://www.zhihu.com/question/64727674
apache和nginx工作原理比较 http://www.server110.com/nginx/201402/6543.html
概念了解:CGI,FastCGI,PHP-CGI与PHP-FPM https://www.awaimai.com/371.html