说到URL重写就不得不提URL重定向.

URL重定向

URL重定向是客户端操作,指示客户端访问另一个地址的资源.这需要往返服务器,并且当客户端对资源发出请求时,返回客户端的重定向URL会出现在浏览器的地址栏中.

将请求重定向到不同的URL时,可指示重定向是永久的还是临时的.如果是永久的,则使用"301"状态码.收到"301"状态码时,客户端可能会缓存.如果是临时的,则使用"302"状态码,以使客户端将来不应存储和重用重定向URL.

示例:

新建一个WebAPI项目;新增一个 TestController 控制器;在 Startup 类的 Configure 方法中增加如下代码:

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
...other codes
var options = new RewriteOptions();
//options.AddRedirect("^redirect/(.*)", "api/test");//默认状态码为 302
options.AddRedirect("^redirect/(.*)", "api/test", 301);
app.UseRewriter(options); app.Run(async context =>
{
//注意重定向和重写URL两种情况下,浏览器地址栏和页面显示的 URL 的区别.
await context.Response.WriteAsync($"URL:{context.Request.Path + context.Request.QueryString}");
}); app.UseMvc();
}

启动该项目,在浏览器地址栏输入 : https://localhost:44303/redirect/123 后,

可以看出,客户端一共请求了两次,浏览器地址栏变成了重定向的URL.

URL重写

URL重写是服务器端操作.重写URL不需要往返服务器,重写的URL也不会返回客户端,也不会出现在浏览器地址栏.

示例:

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
...other codes var options = new RewriteOptions();
options.AddRewrite("^rewrite/(.*)", "api/test", true);//重写URL,false/true 表示如果当前规则适用,是否跳过其他重写规则.
app.UseRewriter(options); app.Run(async context =>
{
//注意重定向和重写URL两种情况下,浏览器地址栏和页面显示的 URL 的区别.
await context.Response.WriteAsync($"URL:{context.Request.Path + context.Request.QueryString}");
}); app.UseMvc();
}
}

在浏览器地址栏输入 : https://localhost:44303/rewrite/1


RewriteOptions 类型提供了很多重写和重定向的方法,除了上述的 AddRedirect , AddRewrite 外,还有两个比较好玩的方法.

1.Add(Action<RewriteContext> applyRule)

接收一个委托.当请求地址符合一个规则后,方法传递的委托便会执行.这里以修改 reContext.Result 的值为例.示例:

            {
RewriteOptions options = new RewriteOptions();
options.AddRewrite("^rewrite*", "test", true).Add(reContext =>
{
reContext.Result = RuleResult.EndResponse;
});
app.UseRewriter(options);
app.Run(async context => { await context.Response.WriteAsync(context.Request.Path + context.Request.QueryString); });
}

只有请求地址符合规则时才会执行  app.Run(async context => { await context.Response.WriteAsync(context.Request.Path + context.Request.QueryString); })  这句代码.

测试图如下:

可以看到,上面的请求不符合规则,页面上什么都没显示;下面的请求符合规则,页面显示出了重写后的请求路径.

2.AddRedirectToHttps

该方法可以将 HTTP 请求重定向到采用 HTTPS 的相同主机和路径.

新建一个 WebAPI 项目,但是不勾选  "为HTTPS配置" 选项.

修改 Configure 方法.

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
} var options = new RewriteOptions();
options.AddRedirectToHttps(statusCode: 301, sslPort: 9527);
app.UseRewriter(options);
app.UseMvc();
}

在浏览器地址栏输入该项目地址 : http://localhost:50588 ,会重定向到 https://localhost:9527 , 如图.

上面所有重写的示例中,重写的规则都是写在代码里面的,而ASP.NET Core 还提供了从文件中读取规则的方式.

新建一个文件夹 Rule ,添加一个 IISUrlRewrite.xml 文件,内容如下:

<rewrite>
<rules>
<rule name="MyIISUrlRewrite" stopProcessing="true">
<match url="^rewrite/(.*)"/>
<!--还 没发现 appendQueryString = false 和 true 的区别-->
<action type="Rewrite" url="api/values/{R:1}" appendQueryString="false"/>
</rule>
</rules>
</rewrite>

修改 Configure 方法:

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
......
var options = new RewriteOptions(); //方法一:
//using (StreamReader sr = File.OpenText(Path.Combine(env.ContentRootPath, @"Rule\IISUrlRewrite.xml")))
//{
// options.AddIISUrlRewrite(sr);
// app.UseRewriter(options);
//} //方法二:
var fileProvider = new PhysicalFileProvider(Path.Combine(env.ContentRootPath, "Rule"));
options.AddIISUrlRewrite(fileProvider, "IISUrlRewrite.xml");
app.UseRewriter(options); app.Run(async context =>
{
await context.Response.WriteAsync(context.Request.Path + context.Request.QueryString);
}); app.UseMvc();
}

图就不上了.

虽然还没用 ASP.NET Core 开发过任何项目,但是我觉得下面这种重写和重定向的方法或许会是用得最多的,因为它足够灵活.

    public class MyRule : IRule
{
//可以自定义构造函数,做一些验证
//public MyRule(string extension, string newPath)
//{
// if (string.IsNullOrEmpty(extension))
// {
// throw new ArgumentException(nameof(extension));
// }
// if (!Regex.IsMatch(extension, @"^rewrite*"))
// {
// throw new ArgumentException("Invalid extension", nameof(extension));
// }
// if (!Regex.IsMatch(newPath, @"(/[A-Za-z0-9]+)+?"))
// {
// throw new ArgumentException("Invalid path", nameof(newPath));
// } // _extension = extension;
// _newPath = new PathString(newPath);
//} //private readonly string _extension;
//private readonly PathString _newPath; private readonly string _extension;
private readonly string _newPath;
public MyRule(string extension, string newPath)
{
_extension = extension;
_newPath = newPath;
} public void ApplyRule(RewriteContext context)
{
HttpRequest request = context.HttpContext.Request;
HttpResponse response = context.HttpContext.Response; //可以重写
request.Path = new PathString(_newPath);
context.Result = RuleResult.SkipRemainingRules; //可以重定向
//if (request.Path.Value.EndsWith(_extension, StringComparison.OrdinalIgnoreCase))
//{
// response.StatusCode = StatusCodes.Status302Found;
// context.Result = RuleResult.EndResponse;
// response.Headers[HeaderNames.Location] = _newPath;
//}
}
}
        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
else
{
app.UseHsts();
} RewriteOptions options = new RewriteOptions();
options.Add(new MyRule("rewrite","/api/test"));
app.UseRewriter(options); app.UseHttpsRedirection();
app.UseMvc();
}
}

ASP.NET Core 2.2 基础知识(四) URL重写中间件的更多相关文章

  1. ASP.NET Core 2.2 基础知识(十四) WebAPI Action返回类型(未完待续)

    要啥自行车,直接看手表 //返回基元类型 public string Get() { return "hello world"; } //返回复杂类型 public Person ...

  2. ASP.NET Core 2.2 基础知识(十八) 托管和部署 概述

    为了方便演示,以 .NET Core 控制台应用程序讲解. 我们新建一个控制台应用程序,安装 "Newtonsoft.Json" Nuget 包,然后右键点击该项目,选择" ...

  3. ASP.NET Core 2.2 基础知识(十二) 发送 HTTP 请求

    可以注册 IHttpClientFactory 并将其用于配置和创建应用中的 HttpClient 实例. 这能带来以下好处: 提供一个中心位置,用于命名和配置逻辑 HttpClient 实例. 例如 ...

  4. ASP.NET Core 2.2 基础知识(六) 配置(内含MySql+EF)

    先上一段代码,了解一下 .NET Core 配置数据的结构. 新建一个 控制台项目,添加一个文件 json.json ,文件内容如下: { "country": "cn& ...

  5. ASP.NET Core 2.2 基础知识(十六) SignalR 概述

    我一直觉得学习的最好方法就是先让程序能够正常运行,才去学习他的原理,剖析他的细节. 就好像这个图: 所以,我们先跟着官方文档,创建一个 SignalR 应用: https://docs.microso ...

  6. ASP.NET Core 2.2 基础知识(十三) WebAPI 概述

    我们先创建一个 WebAPI 项目,看看官方给的模板到底有哪些东西 官方给出的模板: [Route("api/[controller]")] [ApiController] pub ...

  7. ASP.NET Core 2.2 基础知识(十一) ASP.NET Core 模块

    ASP.NET Core 应用与进程内的 HTTP 服务器实现一起运行.该服务器实现侦听 HTTP 请求,并在一系列请求功能被写到 HttpContext 时,将这些请求展现到应用中. ASP.NET ...

  8. ASP.NET Core 2.2 基础知识(十) Web服务器 - Kestrel

    ASP.NET Core 应用与进程内的 HTTP 服务器实现一起运行.该服务器实现侦听 HTTP 请求,并在一系列请求功能被写到 HttpContext 时,将这些请求展现到应用中. ASP.NET ...

  9. ASP.NET Core 2.2 基础知识(九) 使用托管服务实现后台任务

    在 ASP.NET Core 中,后台任务作为托管服务实现.托管服务是一个类,而且必须实现 IHostedService 接口,该接口定义了两个方法: StartAsync(CancellationT ...

随机推荐

  1. 线程--promise furture 同步

    http://www.cnblogs.com/haippy/p/3279565.html std::promise 类介绍 promise 对象可以保存某一类型 T 的值,该值可被 future 对象 ...

  2. nc用法小记

    By francis_hao    Jun 30,2017   ncat:连接和重定向套接字 概要 ncat [OPTIONS...] [hostname] [port]   描述 ncat 是一个集 ...

  3. linux 条件判断式

    1.利用if ...then if [ 判断条件 ];then 指令 fi 实例一 Y/N: #!/bin/bash #Program: # This program shows "Hell ...

  4. [NOI2003] 文本编辑器 (splay)

    复制炸格式了,就不贴题面了 [NOI2003] 文本编辑器 Solution 对于光标的移动,我们只要记录一下现在在哪里就可以了 Insert操作:手动维护中序遍历结果,即每次取中点像线段树一样一样递 ...

  5. 怎么利用idea自带的工具,不需要 重启tomcat或则其他服务,js代码自动生效

    idea中有一个工具:可以直接upload,能让你修改的界面直接可以看到,不需要重启服务. 依次点击的按钮如下: 点击进入的界面这个填的只是一个示例,在各位的电脑上肯定不行,大家依据实际情况填写.

  6. bzoj 2426 【HAOI2010】工程选址 贪心

    [HAOI2010]工厂选址 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 447  Solved: 308[Submit][Status][Disc ...

  7. shell编程 if 注意事项

    read -n 1 -p "Let's go(y or n):" if [ "$REPLY"x = "y"x -o "$REPLY ...

  8. linux下解压,压缩命令大全---方便新手查看

    本文最后讲述如何在打包时排除某些文件 .tar 解包:tar xvf FileName.tar 打包:tar cvf FileName.tar DirName (注:tar是打包,不是压缩!) --- ...

  9. 网络编程:connect函数

    TCP客户用connect函数来建立与TCP服务器的连接: cpp #include<sys/socket.h> int connect(int sockfd, const struct ...

  10. Java并发(4)- synchronized与CAS

    引言 上一篇文章中我们说过,volatile通过lock指令保证了可见性.有序性以及"部分"原子性.但在大部分并发问题中,都需要保证操作的原子性,volatile并不具有该功能,这 ...