所谓URL重写指的是更改当前执行的URL,将其指向另外的URL以继续处理当前请求或重定向到外部URL。在ASP.NET中我们可以使用HttpContext.RewritePath方法,但在.NET Core中它并不存在。下面我我们将学习重写和重定向之间的区别,和何时以及如何在ASP.NET Core 中使用它们。 实际开发中,常见的重写URL场景有如下四种:
前两个非常简单,它们是简单的从一个URL跳转到另一个URL,因为某些内容已经移动,或需要不同权限的用户看到不同内容的状态。这在实际项目中是很常见的,尤其是在已经存在了一段时间并且发生了一些变化的应用程序中。一个不太常见但可以说更有用的用例是用于呈现自定义内容工具的URL转换。 最常见的重写场景是应用程序级别的,比如正在构建应用程序,需要在某个情况下重定向到另一个端点,例如登录和身份验证,点击登录URL,这个URL将登录并作为登录流程的一部分,登录成功后将被重定向到起始页或传入的URL对应的页面中。大多数HTML应用程序和一些需要身份认证的REST服务有一些像这样需要显式重定向的请求。这是一个经典的重定向场景。 上面简单的说了一下重写URL和重定向URL的例子,下面我们就来具体讲解以下重写和重定向。
重写改变了当前请求的路径,通过中间件管道继续处理当前请求的所有现有状态。重写后,注册的任何中间件都会收到新的URL,并使用新路径处理请求的其余部分。所有这一切都是作为一个单一的服务器请求的一部分发生的。
Tip:请求的 URL保持不变,不会更改为重写的 URL。
我肯可以使用如下代码重写URL:
context.Request.Path
重定向是在浏览器中通过一个或HTTP响应头触发一个新的HTTP请求,来在服务器上触发一个新的请求。重定向是对客户端的HTTP标头的响应。重定向的头部如下:
HTTP/1.1 302 Moved
Content-Type: text/html; charset=UTF-8
Location: https://xxxxx.com/aaa
重定向还可用于301 Moved Permanently,让搜索引擎知道老的URL已被弃用。 重定向的代码如下:
context.Response.Redirect()
一般来说,重写比重定向来说往往会更有效,因为它避免了服务器往返。并且重写可以保留请求信息,因此拥有POST或PUT操作具有与其关联的数据,则该数据将保持完整,然而重定向总是由浏览器以GET作为操作重新发出,因此无法保留和使用重定向前的数据。
在ASP.NET Core中执行这个操作,最简单的方法是使用app.Use()内联中间件,可以将其添加到Startup.Configure()方法中。下面的代码展示了如何在中间件中处理重写并操作app.Use():
app.Use(async (context,next) =>
{
var url = context.Request.Path.Value;
if (url.Contains("/home/privacy"))
{
context.Request.Path = "/home/index";
}
await next();
});
上面的代码会拦截每个传入的请求并检查要重写的URL,当它找到时更改context.Request.Path并通过管道的后续部分继续处理,所有后续中间件组件现在都可以看到更新的路径。 同样,我们也可以使用类似的方法进行重定向。因为重定向是个新的轻轻,因此需要终止中间件管道,代码如下:
app.Use(async (context,next) =>
{
var url = context.Request.Path.Value;
if (url.Contains("/home/privacy"))
{
context.Response.Redirect("https://www.xxxx.com")
return;
}
await next();
});
一般来说,除非目标URL包含应用程序以外的URL,否则的话我认为没有充分的理由在中间件中使用重定向。因为它只对外部的、非应用程序URL有用。但是凡事都有特殊情况,当我们需要将重定向作为应用程序/控制器逻辑的一部分时,在这种情况下不能使用重写操作,因为路径已经路由到应用程序端点/控制器方法。
TIP:
对于更复杂的重写和重定向场景,我们还可以使用ASP.NET Core重写中间件。它提供了设置基于 regEx的重写和重定向,以及许多不同的和一些常见的重写操作的能力。
加入需要对许多URL有复杂的规则或需要遵循特定模式来重新路由内容,则使用中间件非常有用。例如下面的代码:
var options = new RewriteOptions()
.AddRedirectToHttpsPermanent();
.AddRedirect("redirect-rule/(.*)", "redirected/$1")
.AddRewrite(@"^rewrite-rule/(\d+)/(\d+)", "rewritten?var1=$1&var2=$2",
skipRemainingRules: true)
app.UseRewriter(options);
ASP.NET Core 中的 URL 重写很简单,只需更改调用context.Request.PathURL即可重写,以路由到某些不同的端点。对于外URL可以context.Response.Redirect(),但要注意的是Response.Redirect()它不会自动修复虚拟路径 ( ~/) 语法。对于简单的用例,使用显式中间件处理重写,对于需要基于规则的评估的更复杂的用例,不需要重新发明轮子,因为ASP.NET Core提供了重写中间件,该中间件使用基于HTTP的重写模块中获得的所有常见正则表达式扩展。