location 匹配规则
匹配顺序
存在多个 location 配置块的情况下,匹配顺序如下:
- (1) 首先,精确匹配
=
- (2) 其次,前缀匹配
^~
- (3) 再次,按文件中书写顺序的 正则匹配
- (4) 然后,匹配不带任何修饰符的 前缀匹配
- (5) 最后,交给
/
通用匹配 - (6) 当有匹配成功时,停止匹配,按当前匹配规则处理请求
注意:前缀匹配,如果有包含关系时,按最大匹配原则执行。
比如存在前缀匹配:location /dir01
与 location /dir01/dir02
,现在有请求 http://localhost/dir01/dir02/file
最终将匹配到 location /dir01/dir02
。
例子,有如下匹配规则:
location = / {
echo "规则A";
}
location = /login {
echo "规则B";
}
location ^~ /static/ {
echo "规则C";
}
location ^~ /static/files {
echo "规则X";
}
location ~ \.(gif|jpg|png|js|css)$ {
echo "规则D";
}
location ~* \.png$ {
echo "规则E";
}
location /img {
echo "规则Y";
location / {
echo "规则F";
}
那么产生的效果如下:
- (1) 访问根目录
/
,比如 将匹配规则 A
。 - (2) 访问
http://localhost/login
将匹配规则 B
,`http://localhost/register` 则匹配 `规则 F`。
- (3) 访问
http://localhost/static/a.html
将匹配规则 C
。 - (4) 访问
http://localhost/static/files/a.exe
将匹配规则 X
。虽然 `规则 C` 也能匹配到,但因为 **最大匹配原则**,最终选中了 `规则 X`。
你可以测试下,去掉`规则 X` ,则当前 URL 会匹配上 `规则 C`。
- (5) 访问
http://localhost/a.gif
,http://localhost/b.jpg
将匹配规则 D
和规则 E
。因为 `规则 D` **顺序优先**,所以 `规则 E` 不起作用。
而 `http://localhost/static/c.png` 则优先匹配到 `规则 C`。
- (6) 访问
http://localhost/a.PNG
则匹配规则 E
,而不会匹配规则 D
,因为规则 E
不区分大小写。 - (7) 访问
http://localhost/img/a.gif
会匹配规则 D
。 (9) 访问
http://localhost/category/id/1111
则最终匹配到规则 F
。因为以上规则都不匹配,这个时候应该是 Nginx 转发请求给后端应用服务器,比如 FastCGI (php),tomcat (jsp),Nginx 作为反向代理服务器存在。
综上,所以在实际使用中,笔者觉得至少有三个匹配规则定义,如下:
第一个必选规则
第二个必选规则
第三个必选规则
rewrite 语法
1、 常用的 flag
- break – 中止 rewrite,不再继续匹配
- redirect – 返回临时重定向的 HTTP 状态 302
- permanent – 返回永久重定向的 HTTP 状态 301
2、下面是可以用来判断的表达式:
-f
和!-f
用来判断是否存在 文件-d
和!-d
用来判断是否存在 目录-e
和!-e
用来判断是否存在 文件或目录-x
和!-x
用来判断文件是否 可执行
3、下面是可以用作判断的全局变量 例:http://localhost:88/test1/test2/test.php?k=v
server {
listen 80;
server_name start.igrow.cn;
index index.html index.php;
root html;
if ($http_host !~ "^star\.igrow\.cn$") {
rewrite ^(.*) http://star.igrow.cn$1 redirect;
}
}
防盗链
location ~* \.(gif|jpg|swf)$ {
valid_referers none blocked start.igrow.cn sta.igrow.cn;
if ($invalid_referer) {
rewrite ^/ http://$host/logo.png;
}
}
location ~* \.(js|css|jpg|jpeg|gif|png|swf)$ {
if (-f $request_filename) {
expires 1h;
break;
}
}
禁止访问某个目录
location ~* \.(txt|doc)$ {
root /data/www/wwwroot/linuxtone/test;