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