Nginx 入門(一)Nginx 配置Web服務器

Nginx 入門(一)Nginx 配置Web服務器

引言

Nginx作為高性能的web和反向代理服務器杨帽,在互聯(lián)網(wǎng)公司應用廣泛。作為一名剛入職的小白姨伤,9月底的時候經(jīng)歷了公司站點的HTTPS改造溃斋,雖然沒有親手配置nginx, 而且一開始看到Nginx配置還是很懵逼的 (為什么本科學校不專門學一下啊分尸?%>_<% 還是怪自己不夠主動去學)锦聊,本文寫給從沒接觸過Nginx的同學,也算是入門箩绍,不會太深入孔庭,有興趣的同學可以買《深入理解Nginx》, (真的要有興趣啊,很厚的一本書)材蛛,個人覺得作為開發(fā)人員知道一些基本配置就行了圆到,沒必要特別深入。

Nginx的安裝

首先介紹一下Nginx的在不同系統(tǒng)下的安裝

# CentOS
yum install nginx;
# Ubuntu
sudo apt-get install nginx;
# Mac
brew install nginx;

本文主要以Mac下的安裝為例仰税。
通過homebrew构资,nginx默認被安裝在/usr/local/Cellar/nginx/目錄下。conf安裝目錄在/usr/local/etc/nginx/nginx.conf
啟動陨簇、熱重啟吐绵、關閉以及測試配置的命令如下:

# 啟動
nginx -s start;
# 重新啟動,熱啟動河绽,修改配置重啟不影響線上
nginx -s reload;
# 關閉
nginx -s stop;
# 修改配置后己单,可以通過下面的命令測試是否有語法錯誤
nginx -t;

在瀏覽器中鍵入http://localhost:8080,即可訪問到nginx的歡迎界面。那么耙饰,為什么會訪問到nginx的歡迎界面的呢纹笼?
不妨打開nginx.conf,一起來分析一下這個文件苟跪。在nginx的配置中用#表示注釋

#user  nobody;
##定義擁有和運行Nginx服務的Linux系統(tǒng)用戶

worker_processes  1;
##定義單進程廷痘。通常將其設成CPU的個數(shù)或者內核數(shù)

#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;
##定義Nginx在哪里打日志


#pid        logs/nginx.pid;
##Nginx寫入主進程ID(PID)

events {
    worker_connections  1024;
    ##通過worker_connections和worker_processes計算maxclients蔓涧。
    ##max_clients = worker_processes * worker_connections
}


http {
    include       mime.types;
    ##在/opt/nginx/conf/mime.types寫的配置將在http模塊中解析
    
    default_type  application/octet-stream;

    #log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
    #                  '$status $body_bytes_sent "$http_referer" '
    #                  '"$http_user_agent" "$http_x_forwarded_for"';

    #access_log  logs/access.log  main;

    sendfile        on;
    ##如果是為了獲取本地存儲的靜態(tài)化文件,sendfile可以加速服務端笋额,但是如果是反向代理元暴,那么該功能就失效了。
    #tcp_nopush     on;
##在 nginx 中兄猩,tcp_nopush 配置和 tcp_nodelay "互斥"茉盏。它可以配置一次發(fā)送數(shù)據(jù)的包大小。也就是說枢冤,它不是按時間累計  0.2 秒后發(fā)送包鸠姨,而是當包累計到一定大小后就發(fā)送。在 nginx 中淹真,tcp_nopush 必須和sendfile 搭配使用讶迁。
    #keepalive_timeout  0;
    keepalive_timeout  65;
    ##設置保持客戶端連接時間

    #gzip  on;
##告訴服務端用gzip壓縮
    server {
      ##如果你想對虛擬主機進行配置,可以在單獨的文件中配置server模塊趟咆,然后include進來
        listen       8080;
     ##告訴Nginx TCP端口添瓷,監(jiān)聽HTTP連接。listen 80; 和 listen *:80;是一樣的
        server_name  localhost;
        ##定義虛擬主機的名字
        #charset koi8-r;

        #access_log  logs/host.access.log  main;

        location / {
        ##location模塊可以配置nginx如何反應資源請求
            root   html;
            index  index.html index.htm;
        }

        #error_page  404              /404.html;

        # redirect server error pages to the static page /50x.html
        #
        error_page   500 502 503 504  /50x.html;
        location = /50x.html {
            root   html;
        }

        # proxy the PHP scripts to Apache listening on 127.0.0.1:80
        #
        #location ~ \.php$ {
        #    proxy_pass   http://127.0.0.1;
        #}

        # pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
        #
        #location ~ \.php$ {
        #    root           html;
        #    fastcgi_pass   127.0.0.1:9000;
        #    fastcgi_index  index.php;
        #    fastcgi_param  SCRIPT_FILENAME  /scripts$fastcgi_script_name;
        #    include        fastcgi_params;
        #}

        # deny access to .htaccess files, if Apache's document root
        # concurs with nginx's one
        #
        #location ~ /\.ht {
        #    deny  all;
        #}
    }


    # another virtual host using mix of IP-, name-, and port-based configuration
    #
    #server {
    #    listen       8000;
    #    listen       somename:8080;
    #    server_name  somename  alias  another.alias;

    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}


    # HTTPS server
    #
    #server {
    #    listen       443 ssl;
    #    server_name  localhost;

    #    ssl_certificate      cert.pem;
    #    ssl_certificate_key  cert.key;

    #    ssl_session_cache    shared:SSL:1m;
    #    ssl_session_timeout  5m;

    #    ssl_ciphers  HIGH:!aNULL:!MD5;
    #    ssl_prefer_server_ciphers  on;

    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}
    include servers/*;
}

雖然上面的默認配置很多值纱,但是可以總體歸納為三個模塊:

 
#全局模塊
events {
    #events模塊
}

http 
{

   #http全局模塊
 
    server 
    {
    
        #server全局模塊
     
        location [PATTERN]{
           #location模塊
        }
    }

}  

1、全局塊:配置影響nginx全局的指令坯汤。一般有運行nginx服務器的用戶組虐唠,nginx進程pid存放路徑,日志存放路徑惰聂,配置文件引入疆偿,允許生成worker process數(shù)等。

2搓幌、events塊:配置影響nginx服務器或與用戶的網(wǎng)絡連接杆故。有每個進程的最大連接數(shù),選取哪種事件驅動模型處理連接請求溉愁,是否允許同時接受多個網(wǎng)路連接处铛,開啟多個網(wǎng)絡連接序列化等。

3拐揭、http塊:可以嵌套多個server撤蟆,配置代理,緩存堂污,日志定義等絕大多數(shù)功能和第三方模塊的配置家肯。如文件引入,mime-type定義盟猖,日志自定義讨衣,是否使用sendfile傳輸文件换棚,連接超時時間,單連接請求數(shù)等反镇。

4固蚤、server塊:配置虛擬主機的相關參數(shù),一個http中可以有多個server愿险。

5颇蜡、location塊:配置請求的路由,以及各種頁面的處理情況辆亏。

Nginx配置Web服務器

先介紹對一個web服務進行簡單配置风秤,然后對各個重要點簡單說明。這個案例中關于反向代理的要點將在下一篇中介紹扮叨。

案列

########### 每個指令必須有分號結束缤弦。#################
#user administrator administrators;  #配置用戶或者組,默認為nobody nobody彻磁。
#worker_processes 2;  #允許生成的進程數(shù)碍沐,默認為1
#pid /nginx/pid/nginx.pid;   #指定nginx進程運行文件存放地址
error_log log/error.log debug;  #制定日志路徑,級別衷蜓。這個設置可以放入全局塊累提,http塊,server塊磁浇,級別以此為:debug|info|notice|warn|error|crit|alert|emerg
events {
    accept_mutex on;   #設置網(wǎng)路連接序列化斋陪,防止驚群現(xiàn)象發(fā)生,默認為on
    multi_accept on;  #設置一個進程是否同時接受多個網(wǎng)絡連接置吓,默認為off
    #use epoll;      #事件驅動模型无虚,select|poll|kqueue|epoll|resig|/dev/poll|eventport
    worker_connections  1024;    #最大連接數(shù),默認為512
}
http {
    include       mime.types;   #文件擴展名與文件類型映射表
    default_type  application/octet-stream; #默認文件類型衍锚,默認為text/plain
    #access_log off; #取消服務日志    
    log_format myFormat '$remote_addr–$remote_user [$time_local] $request $status $body_bytes_sent $http_referer $http_user_agent $http_x_forwarded_for'; #自定義格式
    access_log log/access.log myFormat;  #combined為日志格式的默認值
    sendfile on;   #允許sendfile方式傳輸文件友题,默認為off,可以在http塊戴质,server塊度宦,location塊。
    sendfile_max_chunk 100k;  #每個進程每次調用傳輸數(shù)量不能大于設定的值,默認為0,即不設上限腋寨。
    keepalive_timeout 65;  #連接超時時間顾复,默認為75s,可以在http,server,location塊十艾。

    upstream mysvr {   
      server 127.0.0.1:7878;
      server 192.168.10.121:3333 backup;  #熱備
    }
    error_page 404 https://www.baidu.com; #錯誤頁    
    server {
        keepalive_requests 120; #單連接請求上限次數(shù)漾稀。
        listen       4545;   #監(jiān)聽端口
        server_name  127.0.0.1;   #監(jiān)聽地址       
        location  ~*^.+$ {       #請求的url過濾模闲,正則匹配,~為區(qū)分大小寫崭捍,~*為不區(qū)分大小寫尸折。
           #root path;  #根目錄
           #index vv.txt;  #設置默認頁
           proxy_pass  http://mysvr;  #請求轉向mysvr 定義的服務器列表
           deny 127.0.0.1;  #拒絕的ip
           allow 172.18.5.54; #允許的ip           
        } 
    }
} 

域名與端口配置

上述例子中 listen 4545; #監(jiān)聽端口 表示監(jiān)聽端口是4545。但是對于一個小白來說有時候看到 listen [::]:80;,listen :80;,listen *:80; 這三種寫法還是會很懵逼的殷蛇,那么他們之間有什么區(qū)別笆导小?

listen [::]:80;表示Nginx會同時監(jiān)聽IPv4和IPv6的80端口粒梦,listen :80;,listen *:80; 這兩種寫法是一樣的亮航,

location中URL匹配

上述例子中,大家發(fā)現(xiàn)location 后面跟著的正則匹配匀们,其實在nginx中缴淋,location url 匹配是遵循一定優(yōu)先級的。

location = / {
    # 完全匹配  =
    # 大小寫敏感 ~
    # 忽略大小寫 ~*
}
location ^~ /images/ {
    # 前半部分匹配 ^~
     # 匹配任何以 /images/ 開頭的地址泄朴,匹配符合以后重抖,停止往下搜索正則,采用這一條祖灰。
}
location ~* \.(gif|jpg|jpeg)$ {
    # ~* 表示執(zhí)行一個正則匹配钟沛,不區(qū)分大小寫
    # ~ 表示執(zhí)行一個正則匹配,區(qū)分大小寫
    # 匹配所有以 gif,jpg或jpeg 結尾的請求
}
location / {
    # 如果以上都未匹配局扶,會進入這里
}

location中的優(yōu)先級如下

(location =) > (location 完整路徑) > (location ^~ 路徑) > (location ,* 正則順序) > (location 部分起始路徑) > (/)

location = / {
#僅僅匹配請求
[ configuration A ]
}
location / {
#匹配所有以 / 開頭的請求讹剔。但是如果有更長的同類型的表達式,則選擇更長的表達式详民。如果有正則表達式可以匹配,則優(yōu)先匹配正則表達式陌兑。
[ configuration B ]
}
location /documents/ {
# 匹配所有以 /documents/ 開頭的請求沈跨。但是如果有更長的同類型的表達式,則選擇更長的表達式兔综。
#如果有正則表達式可以匹配饿凛,則優(yōu)先匹配正則表達式。
[ configuration C ]
}
location ^~ /images/ {
# 匹配所有以 /images/ 開頭的表達式软驰,如果匹配成功涧窒,則停止匹配查找。所以锭亏,即便有符合的正則表達式location纠吴,也
# 不會被使用
[ configuration D ]
}
location ~* \.(gif|jpg|jpeg)$ {
# 匹配所有以 gif jpg jpeg結尾的請求。但是 以 /images/開頭的請求慧瘤,將使用 Configuration D
[ configuration E ]
}

文件路徑定義

在location模塊中可以定義文件路徑
比如
根目錄設置:

location / {
    root /home/barret/test/;
}

主頁設置:

index /html/index.html /php/index.php;

try_files 設置
try_file主要是功能是去檢查文件是否存在戴已,使用第一個被找到文件返回固该。如果沒有一個文件找到, 那么重定向到最后一個參數(shù)指定的URI。如:

location /images/ {
    try_files $uri /images/default.gif;
}

location = /images/default.gif {
    expires 30s;
}

ps: $uri 是不帶請求參數(shù)的當前URI糖儡,下面的全局變量中會介紹
,最后一個參數(shù)也可以是命名的location伐坏。如下:

try_files $uri $uri.html $uri/index.html @other;
location @other {
    # 嘗試尋找匹配 uri 的文件,失敗了就會轉到上游處理
    proxy_pass  http://localhost:9000;
}

location / {
    # 嘗試尋找匹配 uri 的文件握联,沒找到直接返回 502
    try_files $uri $uri.html =502;
}

Rewrite 重定向

如果要把一個URL http://www.reibang.com/users/10001 重寫成 http://www.reibang.com/show?user=10001桦沉,可以使用rewrite 規(guī)則,參見下面的代碼金闽。我在公司站點的改造過程中纯露,遇到了rewrite,重寫URL目的是為了更好的SEO呐矾。

location /users/ {
    rewrite ^/users/(.*)$ /show?user=$1 break;
}

rewrite功能就是苔埋,使用nginx提供的全局變量或自己設置的變量,結合正則表達式和標志位實現(xiàn)url重寫以及重定向蜒犯。

rewrite 規(guī)則 定向路徑 重寫類型;

1组橄、規(guī)則:可以是字符串或者正則來表示想匹配的目標url
2、定向路徑:表示匹配到規(guī)則后要定向的路徑罚随,如果規(guī)則里有正則玉工,則可以使用$index來表示正則里的捕獲分組
3、重寫類型:
last :相當于Apache里德(L)標記淘菩,表示完成rewrite遵班,瀏覽器地址欄URL地址不變
break;本條規(guī)則匹配完成后潮改,終止匹配狭郑,不再匹配后面的規(guī)則,瀏覽器地址欄URL地址不變
redirect:返回302臨時重定向汇在,瀏覽器地址會顯示跳轉后的URL地址
permanent:返回301永久重定向翰萨,瀏覽器地址欄會顯示跳轉后的URL地址

break 與 last的區(qū)別

  • last一般寫在server和if中,而break一般使用在location中
  • last不終止重寫后的url匹配糕殉,即新的url會再從server走一遍匹配流程亩鬼,而break終止重寫后的匹配
  • break和last都能組織繼續(xù)執(zhí)行后面的rewrite指令
    在location里一旦返回break則直接生效并停止后續(xù)的匹配location
    舉個例子:
server {
    location / {
        rewrite /last/ /q.html last;
        rewrite /break/ /q.html break;
    }
    location = /q.html {
        return 400;
    }
}
  • 訪問/last/時重寫到/q.html,然后使用新的uri再匹配阿蝶,正好匹配到locatoin = /q.html然后返回了400
  • 訪問/break時重寫到/q.html雳锋,由于返回了break,則直接停止了

if表達式

上面的簡單重寫很多時候滿足不了需求羡洁,比如需要判斷當文件不存在時玷过、當路徑包含xx時等條件,則需要用到if
if的語法如下:

if (表達式) {
}

內置的全局變量:

$args :這個變量等于請求行中的參數(shù),同$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 : 已經(jīng)經(jīng)過Auth Basic Module驗證的用戶名墓懂。
$request_filename : 當前請求的文件路徑,由root或alias指令與URI請求生成霉囚。
$scheme : HTTP方法(如http捕仔,https)。
$server_protocol : 請求使用的協(xié)議盈罐,通常是HTTP/1.0或HTTP/1.1榜跌。
$server_addr : 服務器地址,在完成一次系統(tǒng)調用后可以確定這個值盅粪。
$server_name : 服務器名稱钓葫。
$server_port : 請求到達服務器的端口號。
$request_uri : 包含請求參數(shù)的原始URI票顾,不包含主機名础浮,如:”/foo/bar.php?arg=baz”。
$uri : 不帶請求參數(shù)的當前URI奠骄,$uri不包含主機名豆同,如”/foo/bar.html”。
$document_uri : 與$uri相同含鳞。

內置的條件判斷:

-f和!-f用來判斷是否存在文件
-d和!-d用來判斷是否存在目錄
-e和!-e用來判斷是否存在文件或目錄
-x和!-x用來判斷文件是否可執(zhí)行

有時候在配置文件中看到$http_host诱告。他和$host有什么不同呢?

$http_host$host都是原始的’HOST’字段
比如請求的時候HOST的值是www.csdn.net 那么反代后還是www.csdn.net如果客戶端發(fā)過來的請求的header中沒有有’HOST’這個字段時民晒,
建議使用$host,這時候的$host就等于server_name锄禽。

if 表達式例子:

# 如果文件不存在則返回400
if (!-f $request_filename) {
    return 400;
}
# 如果host不是xuexb.com潜必,則301到xuexb.com中
if ( $host != 'xuexb.com' ){
    rewrite ^/(.*)$ https://xuexb.com/$1 permanent;
}
# 如果請求類型不是POST則返回405
if ($request_method = POST) {
    return 405;
}
# 如果參數(shù)中有 a=1 則301到指定域名
if ($args ~ a=1) {
    rewrite ^ http://example.com/ permanent;
}

if 通常與location規(guī)則搭配使用,如:

# 訪問 /test.html 時
location = /test.html {
    # 默認值為xiaowu
    set $name xiaowu;
    # 如果參數(shù)中有 name=xx 則使用該值
    if ($args ~* name=(\w+?)(&|$)) {
        set $name $1;
    }
    # 301
    rewrite ^ /$name.html permanent;
}

上面表示:

  • /test.html => /xiaowu.html
  • /test.html?name=ok => /ok.html?name=ok

本文主要參考以下這個文獻
http://www.linuxeye.com/configuration/2657.html


感謝閱讀沃但,下一篇文章會介紹Nginx 配置反向代理服務器
下面是些面試資料

https://zfau4tzxp5.feishu.cn/docx/ByM0dxtWoo87QtxnWtTcG0kCnpg
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末磁滚,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌垂攘,老刑警劉巖维雇,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異晒他,居然都是意外死亡吱型,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進店門陨仅,熙熙樓的掌柜王于貴愁眉苦臉地迎上來津滞,“玉大人,你說我怎么就攤上這事灼伤〈バ欤” “怎么了?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵狐赡,是天一觀的道長撞鹉。 經(jīng)常有香客問我,道長颖侄,這世上最難降的妖魔是什么鸟雏? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮发皿,結果婚禮上崔慧,老公的妹妹穿的比我還像新娘。我一直安慰自己穴墅,他們只是感情好惶室,可當我...
    茶點故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著玄货,像睡著了一般皇钞。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上松捉,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天夹界,我揣著相機與錄音,去河邊找鬼隘世。 笑死可柿,一個胖子當著我的面吹牛,可吹牛的內容都是我干的丙者。 我是一名探鬼主播复斥,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼械媒!你這毒婦竟也來了目锭?” 一聲冷哼從身側響起评汰,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎痢虹,沒想到半個月后被去,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡奖唯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年惨缆,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片臭埋。...
    茶點故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡踪央,死狀恐怖,靈堂內的尸體忽然破棺而出瓢阴,到底是詐尸還是另有隱情畅蹂,我是刑警寧澤,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布荣恐,位于F島的核電站液斜,受9級特大地震影響,放射性物質發(fā)生泄漏叠穆。R本人自食惡果不足惜少漆,卻給世界環(huán)境...
    茶點故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望硼被。 院中可真熱鬧示损,春花似錦、人聲如沸嚷硫。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仔掸。三九已至脆贵,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間起暮,已是汗流浹背卖氨。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留负懦,地道東北人筒捺。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像纸厉,于是被迫代替她去往敵國和親焙矛。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內容