《ASP.NET Core 6框架揭秘》实例演示[31]:路由高阶用法

ASP.NET的路由是通过EndpointRoutingMiddleware和EndpointMiddleware这两个中间件协作完成的,它们在ASP.NET平台上具有举足轻重的地位,MVC和gRPC框架,Dapr的Actor和发布订阅编程模式都建立在路由系统之上。Minimal API更是将提升到了前所未有的高度,上一篇通过9个实例演示了基于路由的REST API开发,本篇演示一些“高阶”的用法。

[S2010]解析路由模式 (源代码)
[S2011]利用多个中间件来构建终结点处理器(源代码)
[S2012]在参数上标注特性来决定绑定的数据源(源代码)
[S2013]默认的参数绑定规则(源代码)
[S2014]针对TryPar[Se方法的参数绑定(源代码)
[S2015]针对BindA[Sync方法的参数绑定(源代码)
[S2016]自定义路由约束(源代码)

[S2010]解析路由模式

下面我们通过一个简单的实例演示如何利用RoutePatternFactory对象解析指定的路由模板,并生成对应的RoutePattern对象。我们定义了如下所示的Format方法将指定的RoutePattern对象格式化成一个字符串。

static string Format(RoutePattern pattern)
{
    var builder = new StringBuilder();
    builder.AppendLine(
        $"RawText:{pattern.RawText}");
    builder.AppendLine(
        $"InboundPrecedence:{pattern.InboundPrecedence}");
    builder.AppendLine(
        $"OutboundPrecedence:{pattern.OutboundPrecedence}");
    var segments = pattern.PathSegments;
    builder.AppendLine("Segments");
    foreach (var segment in segments)
    {
        foreach (var part in segment.Parts)
        {
            builder.AppendLine($"\t{ToString(part)}");
        }
    }
    builder.AppendLine("Defaults");
    foreach (var @default in pattern.Defaults)
    {
        builder.AppendLine(
            $"\t{@default.Key} = {@default.Value}");
    }

    builder.AppendLine("ParameterPolicies ");
    foreach (var policy in pattern.ParameterPolicies)
    {
        builder.AppendLine( 
            $"\t{policy.Key} = {string.Join(',',policy.Value.Select(it => it.Content))}");
    }

    builder.AppendLine("RequiredValues");
    foreach (var required in pattern.RequiredValues)
    {
        builder.AppendLine(
            $"\t{required.Key} = {required.Value}");
    }

    return builder.ToString();

    static string ToString(RoutePatternPart part)
        => part switch
        {
            RoutePatternLiteralPart literal 
            => $"Literal: {literal.Content}",
            RoutePatternSeparatorPart separator 
            => $"Separator: {separator.Content}",
            RoutePatternParameterPart parameter 
            => @$"Parameter: Name = {parameter.Name}; Default = {parameter.Default}; IsOptional = { parameter.IsOptional}; IsCatchAll = { parameter.IsCatchAll};ParameterKind = { parameter.ParameterKind}",
            _ => throw new ArgumentException("Invalid RoutePatternPart.")
        };
}

如下的演示程序调用了RoutePatternFactory 类型的静态方法Parse解析指定的路由模板“weather/{city:regex(^0\d{{2,3}}$)=010}/{days:int:range(1,4)=4}/{detailed?}”生成一个RoutePattern对象,我们在调用该方法时还指定了requiredValues参数。我们调用创建的WebApplication对象的MapGet方法注册了针对根路径“/”的终结点,对应的处理器直接返回RoutePattern对象格式化生成的字符串。

using Microsoft.AspNetCore.Routing.Patterns;
using System.Text;

var template =@"weather/{city:regex(^0\d{{2,3}}$)=010}/{days:int:range(1,4)=4}/{detailed?}";
var pattern = RoutePatternFactory.Parse(
    pattern: template,
    defaults: null,
    parameterPolicies: null,
    requiredValues: new { city = "010", days = 4 });

var app = WebApplication.Create();
app.MapGet("/", ()=> Format(pattern));
app.Run();

如果利用浏览器访问启动后的应用程序,回到得到如图1所示结果,它结构化地展示了路由模式的原始文本、出入栈路由匹配权重、每个段的组成、路由参数的默认值和参数策略,以及生成URL必须提供的默认参数值。

5fe1bd9801fd81f8a18156e804c28e1c.png

图1 针对路由模式的解析

[S2011]利用多个中间件来构建终结点处理器

如果某个终结点针对请求处理的逻辑相对复杂,需要多个中间件协同完成,我们可以调用IEndpointRouteBuilder 对象的CreateApplicationBuilder方法创建一个新的IApplicationBuilder对象,并将这些中间件注册到这个该对象上,最后利用它这些中间件转换成RequestDelegate委托。

var app = WebApplication.Create();
IEndpointRouteBuilder routeBuilder = app;
app.MapGet("/foobar", routeBuilder
 .CreateApplicationBuilder()
    .Use(FooMiddleware)
    .Use(BarMiddleware)
    .Use(BazMiddleware)
    .Build());
app.Run();

static async Task FooMiddleware(
    HttpContext context,RequestDelegate next)
{
    await context.Response.WriteAsync("Foo=>");
    await next(context);
};
static async Task BarMiddleware(
    HttpContext context, RequestDelegate next)
{
    await context.Response.WriteAsync("Bar=>");
    await next(context);
};
static Task BazMiddleware(
    HttpContext context, RequestDelegate next) 
    => context.Response.WriteAsync("Baz");

上面的演示程序注册了一个路径模板为“foobar”的路由,并注册了三个中间件来处理路由的请求。该演示程序启动之后,如果我们利用浏览器对路由地址“/foobar”发起请求,将会得到如图2所示的输出结果。呈现出来的字符串是通过注册的三个中间件(FooMiddleware、BarMiddleware和BazMiddleware)输出内容组合而成。

cdf5e233af54d065ac2b639c4a8dc17d.png

图2 输出结果

[S2012]在参数上标注特性来决定绑定的数据源

如下这个演示程序调用WebApplication对象的MapPost方法注册了一个采用“/{foo}”作为模板的终结点。作为终结点处理器的委托指向静态方法Handle,我们为这个方法定义了五个参数,分别标注了上述五个特性。我们将五个参数组合成一个匿名对象作为返回值。

using Microsoft.AspNetCore.Mvc;
var app = WebApplication.Create();
app.MapPost("/{foo}", Handle);
app.Run();

static object Handle(
    [FromRoute] string foo,
    [FromQuery] int bar,
    [FromHeader] string host,
    [FromBody] Point point,
    [FromServices] IHostEnvironment environment)
    => new { Foo = foo, Bar = bar, Host = host, Point = point,
    Environment = environment.EnvironmentName };

public class Point
{
    public int X { get; set; }
    public int Y { get; set; }
}

程序启动之后,我们针对“http://localhost:5000/abc?bar=123”这个URL发送了一个POST请求,请求的主体内容为一个Point对象序列化成生成的JSON。如下所示的是请求报文和响应报文的内容,可以看出Handle方法的foo和bar参数分别绑定的是路由参数“foo”和查询字符串“bar”的值,参数host绑定的是请求的Host报头,参数point是请求主体内容反序列化的结果,参数environment则是由针对当前请求的IServiceProvider对象提供的服务(S2012)。

POST http://localhost:5000/abc?bar=123 HTTP/1.1
Content-Type: application/json
Host: localhost:5000
Content-Length: 18

{"x":123, "y":456}
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8
Date: Sat, 06 Nov 2021 11:55:54 GMT
Server: Kestrel
Content-Length: 100

{"foo":"abc","bar":123,"host":"localhost:5000","point":{"x":123,"y":456},"environment":"Production"}

[S2013]默认的参数绑定规则

如果请求处理器方法的参数没有显式指定绑定数据的来源,路由系统也能根据参数的类型尽可能地从当前HttpContext上下文中提取相应的内容予以绑定。针对如下这几个类型,对应参数的绑定源是明确的。

  • HttpContext:绑定为当前HttpContext上下文。

  • HttpRequest:绑定为当前HttpContext上下文的Request属性。

  • HttpResponse: 绑定为当前HttpContext上下文的Response属性。

  • ClaimsPrincipal: 绑定为当前HttpContext上下文的User属性。

  • CancellationToken: 绑定为当前HttpContext上下文的RequestAborted属性。

上述的绑定规则体现在如下演示程序的调试断言中。这个演示实例还体现了另一个绑定规则,那就是只要当前请求的IServiceProvider能够提供对应的服务,对应参数(“httpContextAccessor”)上标注的FromSerrvicesAttribute特性不是必要的。但是倘若缺少对应的服务注册,请求的主体内容会一般会作为默认的数据来源,所以FromSerrvicesAttribute特性最好还是显式指定为好。对于我们演示的这个例子,如果我们将前面针对AddHttpContextAccessor方法的调用移除,对应参数的绑定自然会失败,但是错误消息并不是我们希望看到的。

using System.Diagnostics;
using System.Security.Claims;

var builder = WebApplication.CreateBuilder();
builder.Services.AddHttpContextAccessor();
var app = builder.Build();
app.MapGet("/", Handle);
app.Run();

static void Handle(
    HttpContext httpContext, 
    HttpRequest request, 
    HttpResponse response,
    ClaimsPrincipal user, 
    CancellationToken cancellationToken, 
    IHttpContextAccessor httpContextAccessor)
{
    var currentContext = httpContextAccessor.HttpContext;
    Debug.Assert(ReferenceEquals(httpContext, currentContext));
    Debug.Assert(ReferenceEquals(request, currentContext.Request));
    Debug.Assert(ReferenceEquals(response, currentContext.Response));
    Debug.Assert(ReferenceEquals(user, currentContext.User));
    Debug.Assert(cancellationToken == currentContext.RequestAborted);
}

[S2014]针对TryParse方法的参数绑定

如果我们在某个类型中定义了一个名为TryParse的静态方法将指定的字符串表达式转换成当前类型的实例,路由系统在对该类型的参数进行绑定的时候会优先从路由参数和查询字符串中提取相应的内容,并通过调用这个方法生成绑定的参数。

var app = WebApplication.Create();
app.MapGet("/", (Point foobar) => foobar);
app.Run();

public class Point
{
    public int X { get; set; }
    public int Y { get; set; }

    public Point(int x, int y)
    {
        X = x;
        Y = y;
    }
    public static bool TryParse(
    string expression, out Point? point)
    {
        var split = expression.Trim('(', ')').Split(',');
        if (split.Length == 2 
            && int.TryParse(split[0], out var x) 
            && int.TryParse(split[1], out var y))
        {
            point = new Point(x, y);
            return true;
        }
        point = null;
        return false;
    }
}

上面的演示程序为自定义的Point类型定义了一个静态的TryParse方法使我们可以将一个以“(x,y)”形式定义的表达式转换成Point对象。注册的终结点处理器委托以该类型为参数,指定的参数名称为“foobar”。我们在发送的请求中以查询字符串的形式提供对应的表达式“(123,456)”,从返回的内容可以看出参数得到了成功绑定。

778865c73c944e5612e668417f26074b.png

图3 TryParse方法针对参数绑定的影响

[S2015]针对BindAsync方法的参数绑定

如果某种类型的参数具有特殊的绑定方式,我们还可以将具体的绑定实现在一个按照约定定义的BindAsync方法中。按照约定,这个BindAsync应该定义成返回类型为ValueTask<T>的静态方法,它可以拥有一个类型为HttpContext的参数,也可以额外提供一个ParameterInfo类型的参数,这两个参数分别与当前HttpContext上下文和描述参数的ParameterInfo对象绑定。前面演示实例中为Point类型定义了一个TryParse方法可以替换成如下这个 BingAsync方法。

public class Point
{
    public int X { get; set; }
    public int Y { get; set; }

    public Point(int x, int y)
    {
        X = x;
        Y = y;
    }

    public static ValueTask<Point?> BindAsync(
        HttpContext httpContext, 
        ParameterInfo parameter)
    {
        Point? point = null;
        var name = parameter.Name;
        var value = httpContext.GetRouteData()
        .Values.TryGetValue(name!, out var v) 
        ? v 
        : httpContext.Request.Query[name!].SingleOrDefault();

        if (value is string expression)
        {
            var split = expression.Trim('(', ')')?.Split(',');
            if (split?.Length == 2 
                && int.TryParse(split[0], out var x)  
                && int.TryParse(split[1], out var y))
            {
                point = new Point(x, y);
            }
        }
        return new ValueTask<Point?>(point);
    }
}

[S2016]自定义路由约束

我们可以使用预定义的IRouteConstraint实现类型完成一些常用的约束,但是在一些对路由参数具有特定约束的应用场景中,我们不得不创建自定义的约束类型。举个例子,如果需要对资源提供针对多语言的支持,最好的方式是在请求的URL中提供对应的Culture。为了确保包含在URL中的是一个合法有效的Culture,最好为此定义相应的约束。下面将通过一个简单的实例来演示如何创建这样一个用于验证Culture的自定义路由约束。我们创建了一个提供基于不同语言资源的API。我们将资源文件作为文本资源进行存储,如图4所示,我们创建了两个资源文件 (Resources.resx和Resources.zh.resx),并定义了一个名为hello的文本资源条目。

70889e2d0bfd510d8d0e3114afda9da5.png

图4 存储文本资源的两个资源文件

如下演示程序中注册了一个模板为“resources/{lang:culture}/{resourceName:required}”的终结点。路由参数“{resourceName}”表示资源条目的名称(比如“hello”),另一个路由参数“{lang}”表示指定的语言,约束表达式名称culture对应的就是我们自定义的针对语言文化的约束类型CultureConstraint。因为这是一个自定义的路由约束,我们通过调用IServiceCollection接口的Configure<TOptions>方法将此约束采用的表达式名称(“culture”)和CultureConstraint类型之间的映射关系添加到RouteOptions配置选项中。

using App;
using App.Properties;
using System.Globalization;

var builder = WebApplication.CreateBuilder();
var template = "resources/{lang:culture}/{resourceName:required}";
builder.Services.Configure<RouteOptions>(
    options => options.ConstraintMap
    .Add("culture", typeof(CultureConstraint)));
var app = builder.Build();
app.MapGet(template, GetResource);
app.Run();

static IResult GetResource(
    string lang, string resourceName)
{
    CultureInfo.CurrentUICulture = new CultureInfo(lang);
    var text = Resources.ResourceManager.GetString(resourceName);
    return string.IsNullOrEmpty(text)
        ? Results.NotFound()
        : Results.Content(text);
}

该终结点的处理方法GetResource定义了两个参数,我们知道它们会自动绑定为同名的路由参数。由于系统自动根据当前线程的UICulture来选择对应的资源文件,我们对CultureInfo类型的CurrentUICulture静态属性进行了设置。如果从资源文件将对应的文本提取出来,我们将创建一个ContentResult对象并返回。应用启动之后,我们可以利用浏览器指定匹配的URL获取对应语言的文本。如图5所示,如果指定一个不合法的语言(如“xx”),将会违反我们自定义的约束,此时就会得到一个状态码为“404 Not Found”的响应。

b9c6a79687f36112024f2133d113d3a5.png

图5 采用相应的URL得到某个资源针对某种语言的内容

我们来看看针对语言文化的路由约束CultureConstraint究竟做了什么。如下面的代码片段所示,我们在Match方法中会试图获取作为语言文化内容的路由参数值,如果存在这样的路由参数,就可以利用它创建一个CultureInfo对象。如果这个CultureInfo对象的EnglishName属性名不以“Unknown Language”字符串作为前缀,我们就认为指定的是合法的语言文件。

public class CultureConstraint : IRouteConstraint
{
    public bool Match(
        HttpContext? httpContext, 
        IRouter? route, 
        string routeKey,
        RouteValueDictionary values, 
        RouteDirection routeDirection)
    {
        try
        {
            if (values.TryGetValue(routeKey, out var value) 
                && value is not null)
            {
                return !new CultureInfo((string)value)
                    .EnglishName.StartsWith("Unknown Language");
            }
            return false;
        }
        catch
        {
            return false;
        }
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ASP.NET Core 6 是微软推出的一款跨平台的开发框架,它秉承了 ASP.NET 的优点,并进行了一系列的改进和升级。该框架具有高性能、可扩展性强以及丰富的功能特性,广泛应用于Web应用程序开发。 在ASP.NET Core 6 框架中,实现PDF下载的方式相对简单。我们可以通过使用第三方库,如iTextSharp或PdfSharp,来生成PDF文件,并通过HTTP响应将其发送给用户进行下载。 首先,我们需要在项目的依赖中添加所需的NuGet包,如iTextSharp或PdfSharp。然后,我们可以通过编写代码来生成PDF文件,例如创建一个PDF文档对象,添加内容、样式和格式,并保存到指定的文件路径中。 接下来,我们可以使用ASP.NET Core的HTTP响应来将生成的PDF文件发送给用户进行下载。通过设置正确的响应头信息,我们可以指定文件的名称、类型和大小,以及告诉浏览器直接下载该文件而不是在浏览器中打开。最后,我们将生成的PDF文件的内容以字节流的形式写入到HTTP响应的输出流中,完成文件下载的过程。 需要注意的是,由于ASP.NET Core 6是跨平台的,所以在使用第三方库之前,我们需要确认其是否与该框架兼容。另外,为了保证性能和安全性,我们还需要对生成的PDF文件进行适当的验证和授权。 总之,ASP.NET Core 6框架可以通过使用第三方库来实现PDF下载功能。我们可以通过生成PDF文件并将其发送给用户进行下载,以提供更好的用户体验和功能扩展。 ### 回答2: ASP.NET Core 6是微软公司推出的下一代跨平台开发框架,具有高性能、高可扩展性和高可靠性的特点,广泛应用于Web应用程序的开发。在ASP.NET Core 6框架中,PDF下载是常见的需求之一。 要实现PDF下载功能,首先需要创建一个PDF生成和下载的控制器方法。在这个方法中,可以使用第三方的PDF生成库(例如iTextSharp)来生成PDF文件,然后通过HTTP响应将生成的PDF文件返回给用户。 在ASP.NET Core 6中,可以使用以下步骤来实现PDF下载: 1. 安装所需的PDF生成库。可以使用NuGet包管理器来安装iTextSharp或其他PDF生成库。 2. 创建一个控制器方法,用于生成和下载PDF文件。这个方法应该接受生成PDF所需的参数,并返回一个ActionResult或FileResult对象。 3. 在控制器方法中,使用PDF生成库按照需求生成PDF文件。可以使用库提供的API来添加文本、图像、表格等内容到PDF中。 4. 将生成的PDF文件保存到本地磁盘上的临时目录中。可以使用System.IO命名空间中的方法来实现。 5. 使用FileResult对象返回生成的PDF文件。可以通过设置文件的MIME类型和文件名来告诉浏览器以下载的方式处理这个文件。 6. 在浏览器中调用控制器方法的URL地址,即可触发PDF文件的下载。 综上所述,ASP.NET Core 6框架可以通过使用第三方的PDF生成库,按照上述步骤实现PDF下载功能。开发人员可以根据具体需求选择合适的PDF生成库,并根据项目的架构和要求进行相应的调整和优化。 ### 回答3: ASP.NET Core 6是微软开发的一款开源跨平台的Web应用程序框架,它具有高度灵活性和可扩展性。ASP.NET Core 6框架揭秘PDF下载可能是指获取关于ASP.NET Core 6框架的详细技术资料或教程的PDF文档。 对于ASP.NET Core 6框架揭秘,我们可以通过以下途径来获取相关的PDF下载资源: 1. 官方文档:ASP.NET Core 6官方文档是最权威的参考资料之一,其中包含了关于该框架的详细介绍、教程和最佳实践等内容。可以通过微软官方网站或者开发者文档网站下载相关的PDF文档。 2. 学术论文或专业书籍:有些学术论文或专业书籍可能专门讨论ASP.NET Core 6框架的内部机制或高级特性,这些资源可以帮助开发人员深入理解框架的工作原理。可以通过学术论文数据库或在线图书商店搜索相关的PDF下载。 3. 开发者社区或技术博客:许多开发者社区或技术博客经常发布有关ASP.NET Core 6框架的教程、案例分析或技术文章,这些资源通常以PDF格式提供。可以通过搜索引擎或专业技术社区查找并下载相关的PDF文档。 总之,要获取ASP.NET Core 6框架揭秘的PDF下载,我们可以通过查阅官方文档、学术论文、专业书籍,或者浏览开发者社区和技术博客等途径,找到与该框架相关的PDF资源。这些资源将帮助开发人员深入了解ASP.NET Core 6框架的内部机制和高级特性,提高开发效率和代码质量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值