【nginx運維基礎(6)】Nginx的Rewrite語法詳解

概述

重寫URL是很是有用的一個功能,由於它可讓你提升搜索引擎閱讀和索引你的網站的能力;並且在你改變了本身的網站結構後,無須要求用戶修改他們的書籤,無需其餘網站修改它們的友情連接;它還能夠提升你的網站的安全性;並且一般會讓你的網站更加便於使用和更專業。php

Nginx Rewrite規則相關指令

Nginx Rewrite規則相關指令有if、rewrite、set、return、break等,其中rewrite是最關鍵的指令。css

Rewrite

重寫,寫在server段或者location段均可,後出現的先應用html

#判斷訪問地址
if  ($remote_addr = 192.168.1.100) { 
  return 403;
}

#判斷訪問的是否ie;
if ($http_user_agent ~ MSIE) {
  rewrite ^.*$ /ie.htm;
  break; #不break會循環重定向(是ie重寫到ie.htm,而後又發現是ie,又重寫到ie.htm...)
}

#跳轉到404
if (!-e $document_root$fastcgi_script_name) {
  rewrite ^.*$ /404.html;
  break;
}

注意: Nginx對配置的格式很是的嚴格,if後面必定要有空格,運算符先後也必需要用空格隔開nginx

If 空格 (條件) {
    重寫模式
}
rewrite的核心仍是正則表達式,其餘的只要知道其語法規則既可

規則參考

~ 爲區分大小寫匹配

~* 爲不區分大小寫匹配

!~和!~*分別爲區分大小寫不匹配及不區分大小寫不匹配

-f和!-f用來判斷是否存在文件

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

-e和!-e用來判斷是否存在文件或目錄

-x和!-x用來判斷文件是否可執行

last 至關於Apache裏的[L]標記,表示完成rewrite,呵呵這應該是最經常使用的

set 設置變量

return  返回狀態碼 

break 終止匹配, 再也不匹配後面的規則

redirect 返回302臨時重定向 地址欄會顯示跳轉後的地址

permanent 返回301永久重定向 地址欄會顯示跳轉後的地址

內置變量參考

$args, 請求中的參數;

$content_length, HTTP請求信息裏的"Content-Length";

$content_type, 請求信息裏的"Content-Type";

$document_root, 針對當前請求的根路徑設置值;

$document_uri, 與$uri相同;

$host, 請求信息中的"Host",若是請求中沒有Host行,則等於設置的服務器名;

$limit_rate, 對鏈接速率的限制;

$request_method, 請求的方法,好比"GET"、"POST"等;

$remote_addr, 客戶端地址;

$remote_port, 客戶端端口號;

$remote_user, 客戶端用戶名,認證用;

$request_filename, 當前請求的文件路徑名

$request_body_file

$request_uri, 請求的URI,帶查詢字符串;

$query_string, 與$args相同;

$scheme, 所用的協議,好比http或者是https,好比rewrite  ^(.+)$  $scheme://example.com$1  redirect;

$server_protocol, 請求的協議版本,"HTTP/1.0"或"HTTP/1.1";

$server_addr, 服務器地址,若是沒有用listen指明服務器地址,使用這個變量將發起一次系統調用以取得地址(形成資源浪費);

$server_name, 請求到達的服務器名;

$server_port, 請求到達的服務器端口號;

$uri, 請求的URI,可能和最初的值有不一樣,好比通過重定向之類的。
以上變量也能夠用打印日誌哦

範例分析

Example1
不存在的文件跳到404.htmlweb

if (!-e $document_root$fastcgi_script_name) {
  rewrite ^.*$ /404.html;
  break;
} 
/*
要加break,以 xx.com/dsafsd.html這個不存在頁面爲例,咱們觀察訪問日誌, 日誌中顯示的訪問路徑,依然是GET /dsafsd.html HTTP/1.1

提示: 服務器內部的rewrite和302跳轉不同.302跳轉url會改變,變成從新http請求404.html, 而內部rewrite, 上下文沒變,
就是說 fastcgi_script_name 仍然是 dsafsd.html,所以會循環重定向.
*/

Example2
在不使用break的狀況下,對ie訪問進行重寫正則表達式

if ($http_user_agent ~* msie) { //若是是ie訪問的話設變量爲1;
  set $isie 1;
}

if ($fastcgi_script_name = ie.html) { //若是訪問的腳本爲ie.html,變量爲0;
  set $isie 0;
}

if ($isie 1) { //綜合起來,若是ie訪問的是ie.html這個腳本就不重寫;
  rewrite ^.*$ ie.html;
}

Example3apache

目錄自動加/安全

if (-d $request_filename){
    rewrite ^/(.*)([^/])$ http://$host/$1$2/ permanent;
}

用(1)匹配最後一個非'/'的字符,而後本身強行再添加一個'/'($2變量後的那個) 服務器

Example4網站

Nginx防盜鏈

location ~* ^.+\.(jpg|jpeg|gif|png|swf|rar|zip|css|js)$ {
    valid_referers none blocked *.nixi8.com nixi8.com localhost 192.168.42.188; #定義none(空,直接訪問),blocked(被防火牆標記過的來路),nixi8.com的二級域名和一級域名,localhost,192.168.42.188
    
    if ($invalid_referer) { # 若是不是上面定義的其中一個
        rewrite ^/ http://www.nixi8.com/none.gif;  # 就重寫到一張gif圖片上;
        return 412;
        break;
    }
    
    access_log   off;  # 關閉日誌,下降服務器的損耗
    root /opt/lampp/htdocs/web; 
    expires 3d; 
    break;
}

Example5

隱藏index.php

apache下只要在全局配置文件中設置了缺省首頁index.php就能實現直接到達index.php,可是nginx目前默認狀況下只能到達index.html而不能訪問到index.php,因此只好rewrite重寫使其支持

if (-f $request_filename) { //使其不隱藏index.php的時候也能訪問到
    expires max;
    break;
}
if (!-e $request_filename) {
    rewrite ^/(.*)$ /index.php/$1 last;
}

  1. /
相關文章
相關標籤/搜索