(注:本文參照 NickChapsas的Attributes get a feature long-overdue in C# 11)
今天看一個泛型特性的例子,這個功能在C#11才受支持。
在asp.net core mvc中,可以給action添加filter,達到攔截作用,實現如下:
public class MyFilter : IAsyncActionFilter
{public async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next){Console.WriteLine("Action前執行");await next();Console.WriteLine("Action后執行");}
}
使用方式,在Action上添加ServiceFilter特性即可,如下:
[ServiceFilter(typeof(MyFilter))]???????
public IEnumerable<WeatherForecast> Get()
{return Enumerable.Range(1, 5).Select(index => new WeatherForecast{Date?=?DateTime.Now.AddDays(index),TemperatureC?=?Random.Shared.Next(-20,?55),Summary?=?Summaries[Random.Shared.Next(Summaries.Length)]}).ToArray();
}
在運行前記得把MyFilter注放到Service容器中:
builder.Services.AddScoped<MyFilter>();
為了支持C#11,在項目文件.csproj中,PropertyGroup中添加一行<LangVersion>preview</LangVersion>
<Project?Sdk="Microsoft.NET.Sdk.Web"><PropertyGroup><TargetFramework>net7.0</TargetFramework><Nullable>enable</Nullable><ImplicitUsings>enable</ImplicitUsings><LangVersion>preview</LangVersion></PropertyGroup><ItemGroup><PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="7.0.0-preview.4.22251.1" /><PackageReference Include="Swashbuckle.AspNetCore" Version="6.3.1" /></ItemGroup>
</Project>
現在就可以定義一個繼承IFilterFactory的特性類了,并且是泛型的。
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = true, Inherited = true)]
public class GSWFilterAttribute<TFilter> : Attribute, IFilterFactory, IOrderedFilter where TFilter : IAsyncActionFilter
{public bool IsReusable { get; set; }public int Order { get; set; }public IFilterMetadata CreateInstance(IServiceProvider serviceProvider){if (serviceProvider != null){var filter = (IFilterMetadata)serviceProvider.GetRequiredService(typeof(TFilter));if (filter is IFilterFactory filterFactory){filter = filterFactory.CreateInstance(serviceProvider);}return filter;}else{throw new ArgumentNullException(nameof(serviceProvider));}}
}
使用時,直接把泛型類型放上就可以了,如下:
[GSWFilter<MyFilter>]
public IEnumerable<WeatherForecast> Get()
{return Enumerable.Range(1, 5).Select(index => new WeatherForecast{ Date = DateTime.Now.AddDays(index),TemperatureC = Random.Shared.Next(-20, 55),Summary = Summaries[Random.Shared.Next(Summaries.Length)]}).ToArray();
}
雖然兩種方法實現的功能是一樣的,但后一種看起來更優雅一些。同時說明一下,如果多個MyFilter功能的過濾器,可以增加Order屬性,如下:
[GSWFilter<MyFilter1>(Order = 2)]
[GSWFilter<MyFilter2>(Order = 1)]
public?IEnumerable<WeatherForecast>?Get()