相关推荐recommended
nginx中斜杠(‘‘)总结
作者:mmseoamin日期:2023-12-11

文章目录

  • 一、Location
    • 1.1 location没有'/'
    • 1.2 location有'/'
    • 二、URL自动加'/'
    • 三、proxy_pass
      • 3.1 不增加目录
      • 3.2 增加目录
      • 四、root和alias
      • 五、实例

        一、Location

        nginx每个location都是一个匹配目录,nginx的策略是:访问请求来时,会对访问地址进行解析,从上到下逐个匹配,匹配上就执行对应location大括号中的策略,并根据策略对请求作出相应。

        依访问地址:http://www.wandouduoduo.com/wddd/index.html为例,nginx配置如下:

        location /wddd/ { 
          proxy_connect_timeout 18000; ##修改成半个小时 
          proxy_send_timeout 18000; 
          proxy_read_timeout 18000; 
          proxy_pass http://127.0.0.1:8080; 
        }
        

        那访问时就会匹配这个location,从而把请求代理转发到本机的8080端口的Tomcat服务中,Tomcat响应后,信息原路返回。

        1.1 location没有’/’

        请求就可以模糊匹配以字符串开头的所有字符串

        1.2 location有’/’

        只能精确匹配字符本身。

        举例: 配置 location /wandou 可以匹配 /wandoudouduo 请求,也可以匹配 /wandou*/duoduo

        等等,只要以 wandou 开头的目录都可以匹配到。而 location /wandou/ 必须精确匹配 /wandou/ 这个目录的请求,

        不能匹配 /wandouduoduo/ 或 /wandou*/duoduo 等请求。

        二、URL自动加’/’

        有时候访问的地址要求后面以 / 结尾,如果用户忘记输入 /,Nginx 就会自动加上 /。

        通过一个例子来演示问题:

        server {
        	listen	80;
        	server_name localhost;
        	location / {
        		root html;
        		index index.html;
        	}
        }
        

        要想访问上述资源,很简单,只需要通过 http://192.168.200.133 直接就能访问,地址后面不需要加 /,但是如果将上述的配置修改为如下内容:

        server {
        	listen	80;
        	server_name localhost;
        	location /frx {
        		root html;
        		index index.html;
        	}
        }
        

        这个时候,要想访问上述资源,按照上述的访问方式,我们可以通过 http://192.168.200.133/frx/ 来访问,但是如果地址后面不加斜杠,如 http://192.168.200.133/frx,页面就会出问题。如果不加斜杠,Nginx 服务器内部会自动做一个 301 的重定向,重定向的地址会有一个指令叫 server_name_in_redirect 来决定重定向的地址:

        • 如果该指令为 on重定向的地址为:http://server_name/目录名/
        • 如果该指令为 off重定向的地址为:http://原URL中的域名/目录名/

          所以就拿刚才的地址来说,访问 http://192.168.200.133/frx 如果不加斜杠,那么按照上述规则:

          • 如果指令 server_name_in_redirect 为 on,则 301 重定向地址变为 http://localhost/frx/,IP 发生改变,地址出现了问题
          • 如果指令 server_name_in_redirect 为 off,则 301 重定向地址变为 http://192.168.200.133/frx/。这个符合我们的期望

            注意 server_name_in_redirect 指令在 Nginx 的 0.8.48 版本之前默认都是 on,之后改成了 off,所以现在我们这个版本不需要考虑这个问题,但是如果是 0.8.48 以前的版本并且 server_name_in_redirect 设置为 on,我们如何通过 Rewrite 来解决这个问题?

            解决方案

            我们可以使用 Rewrite 功能为末尾没有斜杠的 URL 自动添加一个斜杠

            server {
            	listen	80;
            	server_name localhost;
            	server_name_in_redirect on;
            	location /frx {
            		if (-d $request_filename){   # 如果请求的资源目录存在
            			rewrite ^/(.*)([^/])$ http://$host// permanent; #  获取第二个括号的值:/
            		}
            	}
            }
            

            $1 是第一个括号的值,$2 是第二个括号的值。

            三、proxy_pass

            这里将发送 http://192.168.199.27/frx/xu 请求。

            3.1 不增加目录

            案例localtionproxy_pass匹配
            1/frxhttp://192.168.199.27/frx/xu
            2/frx/http://192.168.199.27/frx/xu
            3/frxhttp://192.168.199.27///xu
            4/frx/http://192.168.199.27//xu

            若proxy_pass 后加’/',代表去除掉请求和 location 的匹配的字符串

            不加 ’ / ’ 则追加全部请求到地址后面。

            3.2 增加目录

            案例localtionproxy_pass匹配
            1/frxhttp://192.168.199.27/bing/bing/xu
            2/frx/http://192.168.199.27/bing/bingxu
            3/frxhttp://192.168.199.27/bing//bing//xu
            4/frx/http://192.168.199.27/bing//bing/xu

            简而言之,就是如果proxy_pass后面有目录,有没有’/',Nginx都会将匹配 location 的内容从请求路径中剔除,然后将请求路径剩余的字符串拼接到 proxy_pass 后生成新的请求路径。

            举例:proxy_pass 的 ip:port 后接了字符串,因此将 location 的 /frx/ 从原请求路径 /frx/xu

            中剔除,变为 xu,然后将 xu 拼接到 http://192.168.1.48/bing 后生成了新请求,因此其他地址收到的请求就是

            /bingxu。

            四、root和alias

            这两个指令都可以来指定访问资源的路径,那么这两者之间的区别是什么?

            举例说明

            1. 在/usr/local/nginx/html目录下创建一个 images 目录,并在目录下放入一张图片 mv.png 图片。然后进入配置文件,添加如下内容:
            location /images {
                root /usr/local/nginx/html;
            }
            

            访问图片的路径为:http://192.168.91.200/images/mv.png

            如果是root,则请求为/usr/local/nginx/html/images/mv.png

            location /images {
                alias /usr/local/nginx/html;
            }
            

            如果是alias,再次访问上述地址,页面会出现 404 的错误,查看错误日志会发现是因为地址不对,所以验证了:

            • root 的处理结果是:root 路径 + location 路径,location 路径包括匹配后面的请求,即包括 /mv.png

              /usr/local/nginx/html/images/mv.png

              • alias 的处理结果是:使用 alias 路径替换 location 路径

                /usr/local/nginx/html/mv.png

                需要在 alias 后面路径改为:

                location /images {
                    alias /usr/local/nginx/html/images;
                }
                

                如果 location 路径是以 / 结尾,则 alias 也必须是以 / 结尾,root 没有要求。

                将上述配置修改为:

                location /images/ {
                    alias /usr/local/nginx/html/images;
                }
                

                访问就会出问题,查看错误日志还是路径不对,所以需要把 alias 后面加上 /

                location /images/ {
                    alias /usr/local/nginx/html/images/;
                }
                

                小结:

                • root 的处理结果是: root 路径 + location 路径
                • alias 的处理结果是:使用 alias 路径替换 location 路径
                • alias 是一个目录别名的定义,root 则是最上层目录的含义
                • 如果 location 路径是以 / 结尾,则 alias 也必须是以 / 结尾,root 没有要求
                  • alias 不支持 location 的 =

                五、实例

                通过nginx访问 127.0.0.1/api/test

                1. location和proxy_pass都不加’/’
                location /api {
                    proxy_pass http://127.0.0.1:8888;
                }
                

                实际访问地址为127.0.0.1:8888/api/test

                1. location加proxy_pass不加
                location /api/ {
                    proxy_pass http://127.0.0.1:8888;
                }
                

                实际访问地址为127.0.0.1:8888/api/test

                1. location和proxy_pass都加
                location /api/ {
                    proxy_pass http://127.0.0.1:8888/;
                }
                

                实际访问地址为127.0.0.1:8888/test

                1. location不加proxy_pass加
                location /api {
                    proxy_pass http://127.0.0.1:8888/;
                }
                

                实际访问地址为127.0.0.1:8888//test

                总结:只要是proxy_pass 端口后方加了斜杠的那么 location都会被替换不会加到实际访问路径中,包括( proxy_pass

                http://127.0.0.1:8888/xxx,其实就是有目录的情况)也算。