Nginx首先选定由哪一个虚拟主机来处理请求?
让我们从一个简单的配置(其中全部3个虚拟主机都在端口*:80上监听)开始:
server {
listen 80;
server_name example.org www.example.org;
...
}
server {
listen 80;
server_name example.net www.example.net;
...
}
server {
listen 80;
server_name example.com www.example.com;
...
}
在这个配置中,nginx仅仅检查请求的“Host”头以决定该请求应由哪个虚拟主机来处理。如果Host头没有匹配任意一个虚拟主机,或者请求中根本没有包含Host头,那nginx会将请求分发到定义在此端口上的默认虚拟主机。在以上配置中,第一个被列出的虚拟主机即nginx的默认虚拟主机——这是nginx的默认行为。而且,可以显式地设置某个主机为默认虚拟主机,即在"listen"指令中设置"default_server"参数:
server {
listen 80 default_server;
server_name example.net www.example.net;
...
}
"default_server"参数从0.8.21版开始可用。在之前的版本中,应该使用"default"参数代替。
请注意"default_server"是监听端口的属性,而不是主机名的属性。
一个简单PHP站点配置
server {
listen 80;
server_name example.org www.example.org;
root /data/www;
location / {
index index.html index.php;
}
location ~* .(gif|jpg|png)$ {
expires 30d;
}
location ~ .php$ {
fastcgi_pass localhost:9000;
fastcgi_param SCRIPT_FILENAME
$document_root$fastcgi_script_name;
include fastcgi_params;
}
}
首先,nginx使用前缀匹配找出最准确的location,这一步nginx会忽略location在配置文件出现的顺序。上面的配置中,唯一的前缀匹配location是"/",而且因为它可以匹配任意的请求,所以被作为最后一个选择。接着,nginx继续按照配置中的顺序依次匹配正则表达式的location,匹配到第一个正则表达式后停止搜索。匹配到的location将被使用。如果没有匹配到正则表达式的location,则使用刚刚找到的最准确的前缀匹配的location,现在让我们来看使用上面的配置,请求是怎样被处理的:
- 请求"
/logo.gif
"首先匹配上location "/
",然后匹配上正则表达式"\.(gif|jpg|png)$
"。因此,它将被后者处理。根据"root /data/www
"指令,nginx将请求映射到文件/data/www/logo.gif
",并发送这个文件到客户端。 - 请求"
/index.php
"首先也匹配上location "/
",然后匹配上正则表达式"\.(php)$
"。 因此,它将被后者处理,进而被发送到监听在localhost:9000的FastCGI服务器。fastcgi_param指令将FastCGI的参数SCRIPT_FILENAME
的值设置为"/data/www/index.php
",接着FastCGI服务器执行这个文件。变量$document_root
等于root指令设置的值,变量$fastcgi_script_name
的值是请求的uri,"/index.php
"。 - 请求"
/about.html
"仅能匹配上location "/
",因此,它将使用此location进行处理。根据"root /data/www
"指令,nginx将请求映射到文件"/data/www/about.html
",并发送这个文件到客户端。 - 请求"
/
"的处理更为复杂。它仅能匹配上location "/
",因此,它将使用此location进行处理。然后,index指令使用它的参数和"root /data/www
"指令所组成的文件路径来检测对应的文件是否存在。如果文件/data/www/index.html
不存在,而/data/www/index.php
存在,此指令将执行一次内部重定向到"/index.php
",接着nginx将重新寻找匹配"/index.php
"的location,就好像这次请求是从客户端发过来一样。正如我们之前看到的那样,这个重定向的请求最终交给FastCGI服务器来处理。