1. Rewrite概述
URL重写有利于网站首选域的确定,对于同一资源页面多条路径的301重定向有助于URL权重的集中,和apache等web服务软件一样,rewrite的组要功能是实现RUL地址的重定向。Nginx的rewrite功能需要PCRE软件的支持,即通过perl兼容正则表达式语句进行规则匹配的。默认参数编译nginx就会支持rewrite的模块,但是也必须要PCRE的支持, rewrite是实现URL重写的关键指令,根据regex(正则表达式)部分内容,重定向到replacement,结尾是flag标记。
2.Rewrite使用场景
2.1 Nginx跳转需求的实现方式
使用rewrite进行匹配跳转
使用if匹配全局变量后跳转
使用location匹配再跳转
2.2 rewrite放在server{},if{},location{}段中
location只对域名后边的除去传递参数外的字符串起作用
2.3 对域名或参数字符串
使用if全局变量匹配
使用proxy_pass反向代理
3. Nginx语法介绍
正则表达式:
. : 匹配除换行符以外的任意字符
? : 重复0次或1次
+ : 重复1次或更多次
* : 重复0次或更多次
\d :匹配数字
^ : 匹配字符串的开始
$ : 匹配字符串的介绍
{n} : 重复n次
{n,} : 重复n次或更多次
[c] : 匹配单个字符c
[a-z] : 匹配a-z小写字母的任意一个
小括号()之间匹配的内容,可以在后面通过$1来引用,$2表示的是前面第二个()里的内容
语法:rewrite regex replacement [flag];
PS:
<regex>:匹配正则
<replacement>:跳转后的内容
[flag]:rewrite支持的flag标记
flag标志位:
last : 相当于Apache的[L]标记,表示完成rewrite
break : 停止执行当前虚拟主机的后续rewrite指令集
redirect : 返回302临时重定向,地址栏会显示跳转后的地址
permanent : 返回301永久重定向,地址栏会显示跳转后的地址
last与break的区别:
last: 适用场景,一般写在server和if中
URL匹配,不终止重写后的url匹配
break: 适用场景,一般使用再location中
URL匹配, 终止重写后的url、匹配
location 类型:
location = patt {} [精准匹配]
location patt {} [一般匹配]
location ~ patt {} [正则匹配]
location表达式类型:
~ 表示执行一个正则匹配,区分大小写
~* 表示执行一个正则匹配,不区分大小写
^~ 表示普通字符匹配。使用前缀匹配。如果匹配成功,则不再匹配其他location。
= 进行普通字符精确匹配。也就是完全匹配。
@ 它定义一个命名的 location,使用在内部定向时,例如 error_page, try_files
location优先级说明
在nginx的location和配置中location的顺序没有太大关系。跟location表达式的类型有关。相同类型的表达式,字符串长的会优先匹配。
以下是按优先级排列说明:
=:优先级最高。表示精确匹配,一旦匹配成功,则不再查找其他匹配项。
^~:以字符开头的模糊匹配,一旦匹配成功,则不再查找其他匹配项。
正则表达式类型(~/~*/!~/!~*)的优先级次之。如果有多个location的正则能匹配的话,则使用正则表达式最长的那个。
后面跟路径:常规字符串匹配类型。按前缀匹配。
/:通用匹配,如果没有其他匹配,任何请求都会匹配到它
注意:越是能更多的进行匹配的优先级最高
比较rewrite和location
1、相同点:都能实现跳转
2、不同点:rewrite是在同一域名内更改获取资源的路径
location是对一类路径做控制访问或反向代理,还可以proxy_pass到其他机器
3、rewrite会写在location里,执行顺序
执行server快里面的rewrite指令
执行location匹配
执行选定的location中的rewrite指令
location优先级规则
(location = 完整路径) > (location ^~ 完整路径) > (location ~* 完整路径) > (location ~ 完整路径) > (location 完整路径) > (location /)
4. 实验测试
4.1 实验环境
需要两台虚拟机,一台服务器,一台测试机
4.2 实验分类
4.2.1 基于域名的跳转
[root@server ~]# systemctl stop firewalld
[root@server ~]# systemctl disable firewalld
[root@server ~]# setenforce 0
[root@server ~]# systemctl status firewalld
● firewalld.service - firewalld - dynamic firewall daemon
Loaded: loaded (/usr/lib/systemd/system/firewalld.service; disabled; vendor preset: enabled)
Active: inactive (dead)
Docs: man:firewalld(1)
[root@server ~]# vim /etc/resolv.conf
nameserver 114.114.114.114
[root@server ~]#
[root@server ~]# vim /etc/hosts
192.168.152.130 www.1.com www.2.com
[root@server ~]# mkdir -p /var/log/nginx/
[root@server ~]# vim /usr/local/nginx/conf/nginx.conf
#gzip on;
server {
listen 80;
server_name www.1.com;
#域名修改
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
#取消注释,开启并对日志保存路径进行修改
location / {
#在原有location位置添加内容
if ($host ='www.1.com') {
rewrite^/(.*)$ http:// www.2.com/$1 permanent;
}
root html;
index index.html index.htm;
}
[root@server ~]# nginx -t
nginx: the configuration file /usr/local/nginx/conf/nginx.conf syntax is ok
nginx: configuration file /usr/local/nginx/conf/nginx.conf test is successful
[root@server ~]#
[root@server ~]#
[root@server ~]# systemctl restart nginx
按下回车,自动跳转另一个界面
再者,访问一个错误的界面,跳转出来的也是另一个网址的错误地址
4.2.2 基于IP地址访问的跳转
[root@server ~]# vim /usr/local/nginx/conf/nginx.conf
#gzip on;
server {
listen 80;
server_name www.1.com;
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
set $rewrite true;
if ($remote_addr = "192.168.152.130") {
set $rewrite false;
}
if ($rewrite = true){
rewrite (.+) /weihu.html;
}
location = /weihu.html {
root /var/www/html;
}
location / {
root html;
index index.html index.htm;
}
#error_page 404 /404.html;
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root html;
}
}
[root@server ~]# mkdir -p /var/www/html
[root@server ~]# echo '<h1> this is weihu web! </h1>' > /var/www/html/weihu.html
[root@server ~]#
[root@server ~]# nginx -t
nginx: the configuration file /usr/local/nginx/conf/nginx.conf syntax is ok
nginx: configuration file /usr/local/nginx/conf/nginx.conf test is successful
[root@server ~]# systemctl restart nginx
IP 地址为192.168.152.130实验:
IP 地址为192.168.152.129实验:
4.2.3 基于旧域名跳转到新域名后面加目录
[root@server ~]# mkdir -p /usr/local/nginx/html/bbs/post
[root@server ~]# echo "<h1> this is 1.html </h1>" >> /usr/local/nginx/html/bbs/post/1.html
[root@server ~]#
[root@server ~]# echo "192.168.152.130 bbs.1.com" >> /etc/hosts
[root@server ~]# vim /etc/hosts
127.0.0.1 localhost server localhost4 localhost4.localdomain4
::1 localhost localhost.localdomain localhost6 localhost6.localdomain6
192.168.152.130 bbs.1.com www.1.com www.2.com
[root@server ~]# vim /usr/local/nginx/conf/nginx.conf
#gzip on;
server {
listen 80;
server_name bbs.1.com;
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
location /post {
rewrite (.+) http://www.1.com/bbs$1 permanent;
}
location / {
root html;
index index.html index.htm;
}
4.2.4 基于参数匹配(多余的)的跳转
[root@server html]# vim /usr/local/nginx/conf/nginx.conf
server {
listen 80;
server_name www.1.com;
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
if ($request_uri ~ ^/100-(100|200)-(\d+)\.html$) {
rewrite (.*) http://www.1.com permanent;
}
location / {
root html;
index index.html index.htm;
}
4.2.5 基于目录下所有php结尾的文件跳转
[root@server ~]# vim /usr/local/nginx/conf/nginx.conf
#gzip on;
server {
listen 80;
server_name www.1.com;
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
location ~* /upload/.*\.php$ {
rewrite (.+) http://www.1.com permanent;
}
location / {
root html;
index index.html index.htm;
}
[root@server ~]# nginx -t
[root@server ~]# systemctl restart nginx
[root@server ~]# vim /usr/local/nginx/conf/nginx.conf
#gzip on;
server {
listen 80;
server_name www.1.com;
charset utf-8;
access_log /var/log/nginx/www.1.com-access.log;
location ~* /abc/123.html {
rewrite (.+) http://www.1.com permanent;
}
location / {
root html;
index index.html index.htm;
}
[root@server ~]# nginx -t
[root@server ~]# systemctl restart nginx