天天看點

詳解nginx中location、rewrite用法總結

一、location用法總結

location可以把不同方式的請求,定位到不同的處理方式上.

1.location的用法

location ~* /js/.*/.js

  • 以 = 開頭,表示精确比對;如隻比對根目錄結尾的請求,後面不能帶任何字元串。
  • 以^~ 開頭,表示uri以某個正常字元串開頭,不是正則比對
  • 以~ 開頭,表示區分大小寫的正則比對;
  • 以~* 開頭,表示不區分大小寫的正則比對
  • 以/ 開頭,通用比對, 如果沒有其它比對,任何請求都會比對到

location的比對順序是“先比對正則,再比對普通”。

矯正:location的比對順序其實是“先比對普通,再比對正則”。我這麼說,大家一定會反駁我,因為按“先比對普通,再比對正則”解釋不了大家平時習慣的按“先比對正則,再比對普通”的實踐經驗。這裡我隻能暫時解釋下,造成這種誤解的原因是:正則比對會覆寫普通比對。

2.location用法舉例

location正則寫法:

1、精确比對 / ,主機名後面不能帶任何字元串

location = / {
[ configuration A ]
}
           
詳解nginx中location、rewrite用法總結

2. 所有的位址都以 / 開頭,是以這條規則将最後比對到預設請求

# 但是正則和最長字元串會優先比對
location / {
[ configuration B ]
}
           

例:

location / { 
     proxy_pass http://server_pools;
    } 
#這條規則隻有其他不符合要求才能比對到;将是最後比對到的,比對度最低,上面實作的功能是:比如網站是www.blog.com;後面什麼都不輸入的時候,
其他的規則也不比對的時候,最後交給負載均衡池的伺服器
           
  1. 比對任何以 /documents/ 開頭的位址,比對符合以後,還要繼續往下搜尋
# 隻有後面的正規表達式沒有比對到時,這一條才會采用這一條

location /documents/ {
[ configuration C ]
}
           

例:

location /static/
    {
    rewrite ^ http://www.abc.com ;   
    }
#上面實作的功能:假設網站域名為www.blog.com;那麼配置上面的功能是輸入www.blog.com/static/時,不管static後面是什麼頁面(頁面也可以不存在),
那麼最終會同樣跳轉到www.abc.com這個網站。
           
  1. 比對任何以 /documents/ 開頭的位址,比對符合以後,還要繼續往下搜尋
# 隻有後面的正規表達式沒有比對到時,這一條才會采用這一條
location ~ /documents/Abc {
[ configuration CC ]
}
           
  1. 比對任何以 /images/ 開頭的位址,比對符合以後,停止往下搜尋正則,采用這一條。
location ^~ /images/ {
[ configuration D ]
}
           
  1. 比對所有以 gif,jpg或jpeg 結尾的請求
# 然而,所有請求 /images/ 下的圖檔會被 config D 處理,因為 ^~ 到達不了這一條正則

location ~* \.(gif|jpg|jpeg)$ {
[ configuration E ]
}
           

例:

詳解nginx中location、rewrite用法總結

7.# 字元比對到 /images/,繼續往下,會發現 ^~ 存在

location /images/ {
[ configuration F ]
}
           

8.# 最長字元比對到 /images/abc,繼續往下,會發現 ^~ 存在

# F與G的放置順序是沒有關系的

location /images/abc {
[ configuration G ]
}
           

9.# 隻有去掉 config D 才有效:先最長比對 config G 開頭的位址,繼續往下搜尋,比對到這一條正則,采用

location ~ /images/abc/ {
[ configuration H ]
}
           

順序 no優先級:

(location =) > (location 完整路徑) > (location ^~ 路徑) > (location ,* 正則順序) > (location 部分起始路徑) > (/)上面的比對結果 :

按照上面的location寫法,以下的比對示例成立:
/ -> config A
精确完全比對,即使/index.html也比對不了
/downloads/download.html -> config B
比對B以後,往下沒有任何比對,采用B
/images/1.gif -> configuration D
比對到F,往下比對到D,停止往下
/images/abc/def -> config D
最長比對到G,往下比對D,停止往下
你可以看到 任何以/images/開頭的都會比對到D并停止,FG寫在這裡是沒有任何意義的,H是永遠輪不到的,這裡隻是為了說明比對順序
/documents/document.html -> config C
比對到C,往下沒有任何比對,采用C
/documents/1.jpg -> configuration E
比對到C,往下正則比對到E
/documents/Abc.jpg -> config CC
最長比對到C,往下正則順序比對到CC,不會往下到E
           

3、實際使用建議

是以實際使用中,個人覺得至少有三個比對規則定義,如下:

#直接比對網站根,通過域名通路網站首頁比較頻繁,使用這個會加速處理,官網如是說。
#這裡是直接轉發給後端應用伺服器了,也可以是一個靜态首頁
# 第一個必選規則location = / {
proxy_pass http://tomcat:8080/index
}
# 第二個必選規則是處理靜态檔案請求,這是nginx作為http伺服器的強項
# 有兩種配置模式,目錄比對或字尾比對,任選其一或搭配使用
location ^~ /static/ {
root /webroot/static/;
}
location ~* \.(gif|jpg|jpeg|png|css|js|ico)$ {
root /webroot/res/;
}
#第三個規則就是通用規則,用來轉發動态請求到後端應用伺服器
#非靜态檔案請求就預設是動态請求,自己根據實際把握#畢竟目前的一些架構的流行,帶.php,.jsp字尾的情況很少了
location / {
proxy_pass http://tomcat:8080/
}
http://tengine.taobao.org/book/chapter_02.html
http://nginx.org/en/docs/http/ngx_http_rewrite_module.html
           

二、Rewrite用法總結

1.rewrite的定義

rewrite功能就是使用nginx提供的全局變量或自己設定的變量,結合正規表達式和标志位實作url重寫以及重定向。

rewrite隻能放在 server{}, location{}, if{}中,并且隻能對域名後邊的除去傳遞的參數外的字元串起作用。
例如 http://seanlook.com/a/we/index.php?id=1&u=str 隻對/a/we/index.php重寫。
           

2.rewirte的文法

rewrite regex replacement [flag];

如果相對域名或參數字元串起作用,可以使用全局變量比對,也可以使用proxy_pass反向代理。

從上 表明看rewrite和location功能有點像,都能實作跳轉。主要差別在于rewrite是在同一域名内更改擷取資源的路徑,而location是對一類路徑做控制通路或反向代理,可以proxy_pass到其他機器。

很多情況下rewrite也會寫在location裡,它們的執行順序是:

1 執行server塊的rewrite指令

2 執行location比對

3 執行標明的location中的rewrite指令

如果其中某步URI被重寫,則重新循環執行1-3,直到找到真實存在的檔案;循環超過10次,則傳回500 Internal Server Error錯誤。

flag标志位

  • last : 相當于Apache的[L]标記,表示完成rewrite
  • break : 停止執行目前虛拟主機的後續rewrite指令集
  • redirect : 傳回302臨時重定向,位址欄會顯示跳轉後的位址
  • permanent : 傳回301永久重定向,位址欄會顯示跳轉後的位址

    因為301和302不能簡單的隻傳回狀态碼,還必須有重定向的URL,這就是return指令無法傳回301,302的原因了。這裡 last 和 break 差別有點難以了解:

last一般寫在server和if中,而break一般使用在location中

last不終止重寫後的url比對,即新的url會再從server走一遍比對流程,而break終止重寫後的比對

break和last都能組織繼續執行後面的rewrite指令

3.rewrite常用正則.

  • . : 比對除換行符以外的任意字元
  • ? : 重複0次或1次
    • : 重複1次或更多次
    • : 重複0次或更多次
  • \d :比對數字
  • ^ : 比對字元串的開始
  • $ : 比對字元串的結束
  • {n} : 重複n次
  • {n,} : 重複n次或更多次
  • [c] : 比對單個字元c
  • [a-z] : 比對a-z小寫字母的任意一個

    小括号()之間比對的内容,可以在後面通過$1來引用,$2表示的是前面第二個()裡的内容。正則裡面容易讓人困惑的是\轉義特殊字元。

rewrite執行個體

例1:
http {
# 定義image日志格式
log_format imagelog '[$time_local] ' $image_file ' ' $image_type ' ' $body_bytes_sent ' ' $status;
# 開啟重寫日志
rewrite_log on;
 
server {
root /home/www;
 
location / {
# 重寫規則資訊
error_log logs/rewrite.log notice;
# 注意這裡要用‘'單引号引起來,避免{}
rewrite '^/images/([a-z]{2})/([a-z0-9]{5})/(.*)\.(png|jpg|gif)$' /data?file=$3.$4;
# 注意不能在上面這條規則後面加上“last”參數,否則下面的set指令不會執行
set $image_file $3;
set $image_type $4;
}
 
location /data {
# 指定針對圖檔的日志格式,來分析圖檔類型和大小
access_log logs/images.log mian;
root /data/images;
# 應用前面定義的變量。判斷首先檔案在不在,不在再判斷目錄在不在,如果還不在就跳轉到最後一個url裡
try_files /$arg_file /image404.html;
}
location = /image404.html {
# 圖檔不存在傳回特定的資訊
return 404 "image not found\n";
}
}
 
對形如/images/ef/uh7b3/test.png的請求,重寫到/data?file=test.png,于是比對到location /data,先看/data/images/test.png檔案存不存在,如果存在則正常響應,如果不存在則重寫tryfiles到新的image404 location,直接傳回404狀态碼。

例2:
rewrite ^/images/(.*)_(\d+)x(\d+)\.(png|jpg|gif)$ /resizer/$1.$4?width=$2&height=$3? last;
對形如/images/bla_500x400.jpg的檔案請求,重寫到/resizer/bla.jpg?width=500&height=400位址,并會繼續嘗試比對location。
           

if指令與全局變量

if判斷指令文法

if (condition)

{…}

對給定的條件condition進行判斷。如果為真,大括号内的rewrite指令将被執行,if條件(conditon)可以是如下任何内容:

當表達式隻是一個變量時,如果值為空或任何以0開頭的字元串都會當做false

直接比較變量和内容時,使用=或!=

~ 正規表達式比對

~* 不區分大小寫的比對

!~ 區分大小寫的不比對

-f和!-f 用來判斷是否存在檔案

-d和!-d 用來判斷是否存在目錄

-e和!-e 用來判斷是否存在檔案或目錄

-x和!-x 用來判斷檔案是否可執行

例:

如果使用者裝置為IE浏覽器的時候,重定向
if ($http_user_agent ~ MSIE) {
rewrite ^(.*)$ /msie/$1 break;
} //如果UA包含"MSIE",rewrite請求到/msid/目錄下
 
if ($http_cookie ~* "id=([^;]+)(?:;|$)") {
set $id $1;
} //如果cookie比對正則,設定變量$id等于正則引用部分
 
if ($request_method = POST) {
return 405;
} //如果送出方法為POST,則傳回狀态405(Method not allowed)。return不能傳回301,302
 
if ($slow) {
limit_rate 10k;
} //限速,$slow可以通過 set 指令設定
 
if (!-f $request_filename){
break;
proxy_pass http://127.0.0.1;
} //如果請求的檔案名不存在,則反向代理到localhost 。這裡的break也是停止rewrite檢查
 
if ($args ~ post=140){
rewrite ^ http://example.com/ permanent;
} //如果query string中包含"post=140",永久重定向到example.com
 
location ~* \.(gif|jpg|png|swf|flv)$ {
valid_referers none blocked www.jefflei.comwww.leizhenfang.com;
if ($invalid_referer) {
return 404;
} //防盜鍊
}
           

全局變量

下面是可以用作if判斷的全局變量

  • KaTeX parse error: Expected 'EOF', got '#' at position 8: args : #̲這個變量等于請求行中的參數,同query_string
  • $content_length : 請求頭中的Content-length字段。
  • $content_type : 請求頭中的Content-Type字段。
  • $document_root : 目前請求在root指令中指定的值。
  • $host : 請求主機頭字段,否則為伺服器名稱。
  • $http_user_agent : 用戶端agent資訊
  • $http_cookie : 用戶端cookie資訊
  • $limit_rate : 這個變量可以限制連接配接速率。
  • $request_method : 用戶端請求的動作,通常為GET或POST。
  • $remote_addr : 用戶端的IP位址。
  • $remote_port : 用戶端的端口。
  • $remote_user : 已經經過Auth Basic Module驗證的使用者名。
  • $request_filename : 目前請求的檔案路徑,由root或alias指令與URI請求生成。
  • $scheme : HTTP方法(如http,https)。
  • $server_protocol : 請求使用的協定,通常是HTTP/1.0或HTTP/1.1。
  • $server_addr : 伺服器位址,在完成一次系統調用後可以确定這個值。
  • $server_name : 伺服器名稱。
  • $server_port : 請求到達伺服器的端口号。
  • $request_uri : 包含請求參數的原始URI,不包含主機名,如:”/foo/bar.php?arg=baz”。
  • u r i : 不 帶 請 求 參 數 的 當 前 U R I , uri : 不帶請求參數的目前URI, uri:不帶請求參數的目前URI,uri不包含主機名,如”/foo/bar.html”。
  • d o c u m e n t u r i : 與 document_uri : 與 documentu​ri:與uri相同。
http://localhost:88/test1/test2/test.php
$host:localhost
$server_port:88
$request_uri:http://localhost:88/test1/test2/test.php
$document_uri:/test1/test2/test.php
$document_root:/var/www/html
$request_filename:/var/www/html/test1/test2/test.php
           

繼續閱讀