Nginx 配置中一個不起眼字符 "-" 的巨大作用,失之毫厘謬以千里

原文鏈接:http://adkx.net/wcrni

Nginx 作爲一個輕量級的,高性能的 web 服務軟件,因其佔有內存少,併發能力強的特點,而廣受歡迎和使用。國內很多大型互聯網公司也對 Nginx 很是青睞。像 BAT(百度,阿里和騰訊),TMD(頭條,美團和滴滴) 等等。使用過 Nginx 的同學都知道,你只需要按需求準確的更改好配置啓動,那麼就可以優雅的訪問它了。所以說 Nginx 對配置文件的很是看中呢,這就要求我們更改配置文件時一定要再三確認,要不然可能因爲疏忽而引發慘案呢?真實案例,就因爲在配置時,少寫了一個字符 “/”,就造成訪問不通報錯,因而接到投訴。那麼是怎麼引起的呢?原因就是:Nginx 在配置 proxy_pass 代理轉接時,少些 “/” 字符造成的。有同學就有疑問,加不加 “/”, 區別真的那麼大嗎?我們帶着這個疑問,來探究下這個問題。

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, 從而把請求代理轉發到本機的 8080Tomcat 服務中,Tomcat 相應後,信息原路返回。總結:location 如果沒有 “/” 時,請求就可以模糊匹配以字符串開頭的所有字符串,而有 “/” 時,只能精確匹配字符本身。

下面舉個例子說明:

配置 location /wandou 可以匹配 / wandoudouduo 請求,也可以匹配 / wandou*/duoduo 等等,只要以 wandou 開頭的目錄都可以匹配到。而 location /wandou / 必須精確匹配 / wandou / 這個目錄的請求, 不能匹配 / wandouduoduo / 或 / wandou*/duoduo 等請求。

proxy_pass 有無 “/” 的四種區別探究

訪問地址都是以:http://www.wandouduoduo.com/wddd/index.html 爲例。請求都匹配目錄 / wddd/

第一種:加 "/"

   location  /wddd/ {
    proxy_pass  http://127.0.0.1:8080/;
   }

測試結果,請求被代理跳轉到:http://127.0.0.1:8080/index.html

第二種: 不加 "/"

location  /wddd/ {
    proxy_pass http://127.0.0.1:8080;
   }

測試結果,請求被代理跳轉到:http://127.0.0.1:8080/wddd/index.html

第三種: 增加目錄加 "/"

location  /wddd/ {
    proxy_pass http://127.0.0.1:8080/sun/;
   }

測試結果,請求被代理跳轉到:http://127.0.0.1:8080/sun/index.html

第四種:增加目錄不加 "/"

location  /wddd/ {
    proxy_pass http://127.0.0.1:8080/sun;
   }
測試結果,請求被代理跳轉到:http://127.0.0.1:8080/sunindex.html



總結

location 目錄後加 "/", 只能匹配目錄,不加 “/” 不僅可以匹配目錄還對目錄進行模糊匹配。而 proxy_pass 無論加不加“/”, 代理跳轉地址都直接拼接。

爲了加深大家印象可以用下面的配置實驗測試下:

server {
  listen 80;
  server_name localhost;
  
  # http://localhost/wddd01/xxx -> http://localhost:8080/wddd01/xxx
  location /wddd01/ {
    proxy_pass http://localhost:8080;
  }

  # http://localhost/wddd02/xxx -> http://localhost:8080/xxx
  location /wddd02/ {
    proxy_pass http://localhost:8080/;
  }

  # http://localhost/wddd03/xxx -> http://localhost:8080/wddd03*/xxx
  location /wddd03 {
    proxy_pass http://localhost:8080;
  }

  # http://localhost/wddd04/xxx -> http://localhost:8080//xxx,請注意這裏的雙斜線,好好分析一下。
  location /wddd04 {
    proxy_pass http://localhost:8080/;
  }

  # http://localhost/wddd05/xxx -> http://localhost:8080/hahaxxx,請注意這裏的haha和xxx之間沒有斜槓,分析一下原因。
  location /wddd05/ {
    proxy_pass http://localhost:8080/haha;
  }

  # http://localhost/api6/xxx -> http://localhost:8080/haha/xxx
  location /wddd06/ {
    proxy_pass http://localhost:8080/haha/;
  }

  # http://localhost/wddd07/xxx -> http://localhost:8080/haha/xxx
  location /wddd07 {
    proxy_pass http://localhost:8080/haha;
  }
        
  # http://localhost/wddd08/xxx -> http://localhost:8080/haha//xxx,請注意這裏的雙斜槓。
  location /wddd08 {
    proxy_pass http://localhost:8080/haha/;
  }
}

官方站點:www.linuxprobe.com

Linux 命令大全:www.linuxcool.com

劉遄老師 QQ:5604922

Linux 技術交流羣:193666693

本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/lFZBBROktcBvqwDcbhDt5A