实用篇-无处不在的Location

location配置是nginx模块化配置中最出色的一个设计,几乎所有nginx的业务场景都要通过书写多个location配置来顺应业务需要。语法配置和执行规则都相对比较简单,完全可以掌握在脑海之中。

1、配置规则

只要在网上搜一下,或者去官网翻一下文档,都能知道有几种规则,这是官网的:

Syntax:    location [ = | ~ | ~* | ^~ ] uri { ... }location @name { ... }
Default:    —
Context:    server, location

总而言之,包含两部分内容,一部分是匹配前提条件或者叫要求,另外一部分是要匹配的东西,也就是匹配内容。

下面这段是官网说明的谷歌翻译:

在对“%XX”形式编码的文本进行解码之后,针对归一化的URI执行匹配,解析对相对路径组件“.”和“..”的引用,以及将两个或多个相邻斜杠可能压缩为单个斜杠。

localtion可以由前缀字符串或正则表达式定义。正则表达式使用前面的“〜*”修饰符(不区分大小写匹配)或“〜”修饰符(用于区分大小写匹配)指定。要找到匹配给定请求的位置,nginx首先检查使用前缀字符串(前缀位置)定义的位置。其中,选择并记住具有最长匹配前缀的位置。然后检查正则表达式,按照它们在配置文件中的显示顺序。正则表达式的搜索在第一个匹配中终止,并且使用相应的配置。如果没有找到与正则表达式匹配,则使用之前记住的前缀位置的配置。

localtion blocks可以嵌套,下面提到一些例外。

对于不区分大小写的操作系统,如macOS和Cygwin,与前缀字符串的匹配忽略了一个情况(0.7.7)。但是,比较仅限于一个字节的区域设置。

正则表达式可以包含稍后可以在其他指令中使用的捕获(0.7.40)。

如果最长匹配的前缀位置具有“^〜”修饰符,则不会检查正则表达式。

另外,使用“=”修饰符可以定义一个完全匹配的URI和位置。如果找到完全匹配,则搜索终止。例如,如果频繁出现“/”请求,则定义“location = /”将加快对这些请求的处理,因为搜索在第一次比较之后立即终止。这样的位置不能明显地包含嵌套的位置。

在从0.7.1到0.8.41的版本中,如果请求匹配前缀位置而没有“=”和“^〜”修饰符,则搜索也将被终止,而正则表达式未被检查

根据上面的说明可知,默认情况, nginx先检查前缀字符串,然后检查正则表达式,如果前缀字符串匹配到了,并且前缀字符串有这个“^~” 要求,就不配正则了;如果没有这个“^~” ,即使前缀匹配到了,也要去匹配正则表则,如果正则表达式匹配到了,就是用正则表达式的,没有就是用前缀字符串匹配到的路径;

2、无正则表达式匹配

规则:使用匹配到的最长的前缀的路径。

        location  / {
            root   html;
            index  index.html index.htm;
        }        
        location  /img {
            root   D:/nginx/img;
            index  test.png;
        }

这是nginx下载安装完成后,增加了一个叫img的配置,访问路径是http://localhost/img,访问一下看看什么效果,我开启了日志,就不截图了,直接看日志里,nginx查找的路径:

D:/nginx/img/img/test.png" is not found (3: The system cannot find the path specified)

文件查找的路径是D:/nginx/img(这是我配置的路径),下面的img文件下的test.png文件

如果改成这样:

 location  / {
            root   html;
            index  index.html index.htm;
        }        
        location  /img/ {
            root   D:/nginx/img;
            index  test.png;
        }

看看有没有区别,重新启动,在访问http://localhost/img试一下:

"D:/nginx/img/img/test.png" is not found (3: The system cannot find the path specified)

和之前没什么区别,也就是两个效果是一样的,在浏览其中http://localhost/img和http://localhost/img/是一样的。

再改一下,重启:

     location  / {
                root   html;
                index  index.html index.htm;
            }            
            location  /img/ {
                root   D:/nginx/img/;
                index  test.png;
            }

我在重新访问一下:

"D:/nginx/img/img/test.png" is not found (3: The system cannot find the path specified)

还是一样的效果。 所以如果是前缀字符串匹配,就不要纠结后面要不要加“/”的问题。 另外从结果上看,确实符合规则,使用前缀最长的匹配路径。

2、有正则匹配,无符号修饰

规则:无“^~”, 最终使用正则匹配到的路径,正则匹配不到,使用字符串前缀匹配最长的

增加一个配置:

    location  \/img\/ {
                root   D:/nginx/reg/;
                index  test.png;
            }

重启访问:

"D:/nginx/img/img/test.png" is not found (3: The system cannot find the path specified)

结果发现,找的还是D:/nginx/img的img文件夹的文件,好像正则没有匹配上,改一下:

    location  \/img {
                root   D:/nginx/reg/;
                index  test.png;
            }

这下一定可以匹配上了,重启http://localhost/img测试:

"D:/nginx/img/img/test.png" is not found (3: The system cannot find the path specified)

无语了,为什么没有按照规则来呢,为什么没有查找D:/nginx/reg/路径下的文件呢?

回到前面看一下官网说明,发现正则表达式需要"~"或者"~*"指定才行,修改为下面的配置:

    location ~* \/img\/ {                root   D:/nginx/reg/;                index  test.png;
            }

重启访问http://localhost/img测试:

"D:/nginx/reg/img/test.png"

这下按照规则了,看东西看来还是要认真看啊!

改一下正则表达式:

        location ~* \/img {                root   D:/nginx/reg/;                index  test.png;
            }

再重启测试:

"D:/nginx/reg/img/test.png" is not found (3: The system cannot find the path specified)

也可以正常匹配,也就是说正则匹配的情况下,最后一个/要不要都行,效果是一样的

另外“~*” 不区分大小写和“~”就不用试,应该都知道

3、精确匹配=号

规则:精确匹配某个路径,优先级最高

增加一种配置:

    location = /img/test.png {
                root   D:/nginx/denghao/;
                index  test.png;
            }

重启访问路径http://localhost/img/test.png:

"D:/nginx/denghao/img/test.png" failed (3: The system cannot find the path specified)

从结果看出,和规则一致;

改一下配置:

    location = /img/ {
                    root   D:/nginx/wuhouzhui/;
                    index  test.png;
                }

测试结果:

"D:/nginx/reg/img/test.png" is not found (3: The system cannot find the path specified)

从结果上看,最终使用的匹配路径是正则,至于 = /img/,有没有匹配到,我也不清楚,总而言之,这种情况下使用的是正则的路径,有知道朋友解释一下=号有没有匹配到。

4、提高前缀字符串的优先级的“^~”

规则:如果最长匹配的前缀位置具有“^〜”修饰符,则不会检查正则表达式

因为先搜索匹配的是前缀字符串,所有匹配到了,有这个修饰符就不检查正则了,所以正则就不考虑了。 把配置改一下:

            location ^~ /img/ {                root   D:/nginx/img/;                index  test.png;
            }            
            location = /img/test.png {                root   D:/nginx/denghao/;                index  test.png;
            }

访问http://localhost/img/test.png测试:

"D:/nginx/denghao/img/test.png" failed (3: The system cannot find the path specified)

可以看到匹配结果是等号的路径,所以=号的优先级比^~高

5、总结

搜索优先级:

精确匹配 > 字符串匹配( 长 > 短 [ 注: ^~ 匹配则停止匹配 ]) > 正则匹配( 上 > 下 )

使用优先级:

精确匹配 > (^~) > 正则匹配( 上 > 下 )>字符串(长 > 端)

原文发布于微信公众号 - nginx(nginx-study)

原文发表时间:2018-04-22

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏青玉伏案

窥探Swift编程之错误处理与异常抛出

在Swift 2.0版本中,Swift语言对其错误处理进行了新的设计,当然了,重新设计后的结果使得该错误处理系统用起来更爽。今天博客的主题就是系统的搞一下Swi...

21650
来自专栏菩提树下的杨过

bash/shell编程学习(1)

1)定义变量 myvar=abc #注:等号前后不能加空格 #或 myvar="abc" #或 myvar='abc' #注:如果变量后面的值中间本身没有空格,...

20890
来自专栏编程

Python面向对象6:​isinstance、super、有序字典

1、isinstance(对象名,类),查看对象是不是类的实例。 案例1:isinstance(对象名,类) classFoo: print('hh') obj...

20480
来自专栏Python小屋

Python多线程编程基础2:如何创建线程

Python标准库threading中的Thread类用来创建和管理线程对象,支持使用两种方法来创建线程:1)直接使用Thread类实例化一个线程对象并传递一个...

27640
来自专栏微信公众号:Java团长

Java虚拟机体系结构

  众所周知,Java支持平台无关性、安全性和网络移动性。而Java平台由Java虚拟机和Java核心类所构成,它为纯Java程序提供了统一的编程接口,而不管下...

13020
来自专栏开源优测

移动测试Appium之API手册

移动测试Appium之API手册 前言 本文对Appium Python Client中webdriver.py代码进行分析说明。 笔者使用python3.6版...

46290
来自专栏Python小屋

Python面向对象程序设计中属性的作用与用法

公开的数据成员可以在外部随意访问和修改,很难保证用户进行修改时提供新数据的合法性,数据很容易被破坏,并且也不符合类的封装性要求。解决这一问题的常用方法是定义私有...

26240
来自专栏Java学习网

Java同步问题面试知识学习

Java同步问题面试知识学习 同步 在多线程程序中,同步修饰符用来控制对临界区代码的访问。其中一种方式是用synchronized关键字来保证代码的线程安...

27870
来自专栏xingoo, 一个梦想做发明家的程序员

数字按照不同格式转换成字符串

  如果自己写函数,不使用itoa怎么判断呢?   我们用通常的办法,对数字进行每位的除商,得到后与字符'0'相加。 flag = 0; ...

217100
来自专栏python3

python列表

"C:Program Files (x86)python3.6python.exe" D:/python3_study/list1.py

12930

扫码关注云+社区

领取腾讯云代金券