语法介绍
location [=|~|~*|^~|@] uri { ... }
location @name { ... }
一个 location 关键字,后面跟着可选的修饰符(就是[]中间的正则),后面是要匹配的字符,花括号中是要执行的操作。
- =:表示精确匹配
- ~:表示区分大小写正则匹配
- ~*:表示不区分大小写正则匹配
- ^~:表示 URI 以某个常规字符串开头
- !~:表示区分大小写正则不匹配
- !~*:表示不区分大小写正则不匹配
- /:通用匹配,任何请求都会匹配到
匹配顺序
多个 location 配置的情况下,匹配顺序如下:
1、首先精确匹配=
location = /abcd {
[…]
}
-
website.com/abcd
:匹配 -
website.com/ABCD
:可能会匹配,也可以不匹配,取决于操作系统的文件系统是否大小写敏感。Mac 默认是大小写不敏感,Windows 不区分大小,Linux 区分大小写。 -
website.com/abcd?param1¶m2
:匹配,忽略 querystring -
website.com/abcd/
:不匹配,带有结尾的/ -
website.com/abcde
:不匹配
所以经常请求/
的话,可以使用=
来定义 location。
2、其次前缀匹配^~
如果该 location 是最佳的匹配,那么对于匹配这个 location 的字符串,立刻停止后续的正则搜索。注意,这不是一个正则表达式匹配,它的目的是优先于正则表达式的匹配。
3、接着是按文件中顺序的正则匹配(regular expression),如~
或~*
location ~ ^/abcd$ {
[…]
}
^/abcd$
这个正则表达式表示字符串必须以/
开始,以$
结束,中间必须是abcd
-
website.com/abcd
:匹配(完全匹配) -
website.com/ABCD
:不匹配,大小写敏感 -
website.com/abcd?param1¶m2
:匹配 -
website.com/abcd/
:不匹配,不能匹配正则表达式 -
website.com/abcde
:不匹配,不能匹配正则表达式
4、最后是匹配不带任何修饰的前缀匹配(prefix string)
检查使用前缀字符串的 locations,在使用前缀字符串的 locations 中选择最长匹配的来匹配。
server {
location /doc {
[ configuration A ]
}
location /docu {
[ configuration B ]
}
}
/document
能匹配上面 2 个,但前缀字符串顺序不重要,按照匹配长度来确定,所以最终匹配到B
。
小结,匹配顺序就是先=
再^~
,然后是正则,最后是前缀字符串匹配。如果上述规则不好理解,可以看下面的伪代码
function match(uri):
rv = NULL
if uri in exact_match:
return exact_match[uri]
if uri in prefix_match:
if prefix_match[uri] is '^~':
return prefix_match[uri]
else:
rv = prefix_match[uri] // 注意这里没有 return,且这里是最长匹配
if uri in regex_match:
return regex_match[uri] // 按文件中顺序,找到即返回
return rv
@name 的用法
@用来定义一个命名 location。主要用于内部重定向,不能用来处理正常的请求。其用法如下:
location / {
try_files $uri $uri/ @custom
}
location @custom {
# ...do something
}
上例中,当尝试访问 url 找不到对应的文件就重定向到我们自定义的命名 location(此处为 custom)。
注意,命名 location 中不能再嵌套其它的命名 location。
常用配置规则
1、精确匹配
# 将所有请求直接转发给服务器的9090端口
location = / {
proxy_pass http://127.0.0.1:9090/;
}
2、处理静态文件
# 目录匹配
location ^~ /static/ {
root /webroot/static/;
}
# 后缀匹配
location ~* .(gif|jpg|jpeg|png|css|js|ico)$ {
root /webroot/res/;
}
3、转发动态请求到后端应用服务器
# 将/account/开始的请求转发给Account服务器
location /account/ {
proxy_pass http://127.0.0.1:8080/
}
# 将/order/开始的请求转发给Order服务器
location /order/ {
proxy_pass http://127.0.0.1:9090/
}
URL 尾部的/需不需要
URL 尾部是否需要带/,也是我入门 nginx 经常搞错的。
1、location 中的字符有无/,都没有影响
也就是说/user/
和/user
是一样的,除非 location 中带了$对 url 有要求,可参考前面例子。
2、URL 结构是https://domain.com/
的形式,尾部有没有/都不会造成重定向。
因为浏览器在发起请求的时候,默认加上了/,不过很多浏览器在地址栏里也不会显示/
3、如果 URL 的结构是https://domain.com/some-dir/
,尾部如果缺少/将导致重定向。
因为根据约定,URL 尾部的/表示目录,没有/表示文件。所以访问/some-dir/时,服务器会自动去该目录下找对应的默认文件。如果访问/some-dir 的话,服务器会先去找 some-dir 文件,找不到的话会将 some-dir 当成目录,重定向到/some-dir/,去该目录下找默认文件。
root 与 alias
nginx 指定文件路径有两种方式 root 和 alias,主要区别在于 nginx 如何解释 location 后面的 uri,这会使两者分别以不同的方式将请求映射到服务器文件上。它们的使用方法和作用域:
[root]
语法:root path
默认值:root html
配置段:http、server、location、if
处理结果:root 路径+ location 路径
[alias]
语法:alias path
配置段:location
处理结果:使用 alias 路径替换 location 路径
如果一个请求的 URI 是/t/a.html 时,它们表现如下:
# 返回/www/root/html/t/a.html的文件
location ^~ /t/ {
root /www/root/html/;
}
# 返回/www/root/html/new_t/a.html的文件
# 把location后面配置的路径丢弃掉,把当前匹配到的目录指向到指定的目录。
location ^~ /t/ {
alias /www/root/html/new_t/;
}
可以看出 alias 是一个目录别名的定义,root 则是最上层目录的定义。另外 alias 后面必须要用/
结束,否则会找不到文件的,而 root 则可有可无。
那如果 server 和 location 中都出现 root,是怎样的优先级呢??
http {
server {
listen 80;
server_name www.abc.com;
root /home/www/website/;
location / {
root /home/www/ts/;
index index.html;
}
}
}
简单的来说是就近原则,如果 location 中能匹配到,就是用 location 中的 root 配置,忽略 server 中的 root,当 location 中匹配不到的时候,则使用 server 中的 root 配置。