小男孩‘自慰网亚洲一区二区,亚洲一级在线播放毛片,亚洲中文字幕av每天更新,黄aⅴ永久免费无码,91成人午夜在线精品,色网站免费在线观看,亚洲欧洲wwwww在线观看

分享

【asp.net core 系列】10 實戰(zhàn)之ActionFilter

 小世界的野孩子 2021-03-27

0.前言

在上一篇中,我們提到了如何創(chuàng)建一個UnitOfWork并通過ActionFilter設(shè)置啟用。這一篇我們將簡單介紹一下ActionFilter以及如何利用ActionFilter,順便補齊一下上一篇的工具類。

1. ActionFilter 介紹

ActionFilter全稱是ActionFilterAttribute,我們根據(jù)微軟的命名規(guī)范可以看出這是一個特性類,看一下它的聲明:

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = true, Inherited = true)]
public abstract class ActionFilterAttribute : Attribute, IActionFilter, IFilterMetadata, IAsyncActionFilter, IAsyncResultFilter, IOrderedFilter, IResultFilter

這是一個允許標(biāo)注在類和方法上的特性類,允許多個標(biāo)記,標(biāo)注之后子類會繼承父類的特性。然后,這個類是一個抽象類,所以我們可以通過繼承ActionFilterAttribute來編寫自己的ActionFilter。

1.1 ActionFilter的四個方法

對于一個ActionFilter而言,最重要的是它的四個方法:

public virtual void OnActionExecuted(ActionExecutedContext context);
public virtual void OnActionExecuting(ActionExecutingContext context);

public virtual void OnResultExecuted(ResultExecutedContext context);
public virtual void OnResultExecuting(ResultExecutingContext context);

image-20200615231334442

上圖是這四個方法在一次請求中執(zhí)行的順序。在一次請求真正執(zhí)行之前,想要攔截這個請求,應(yīng)該使用OnActionExecuting。

為什么單獨說這個呢?因為這個方法的出鏡率很高,大多數(shù)時候都會使用這個方法進行請求過濾。

1.2 在ActionFilter中我們能做什么

我們來簡單介紹一下,四個方法中的四種上下文類型,看一看里面有哪些我們可以利用的方法:

1.2.1 ActionExecutingContext

這是一個Action執(zhí)行前的上下文,表示Action并未開始執(zhí)行,但是已經(jīng)獲取到了控制器實例:

public class ActionExecutingContext : FilterContext
{
    public virtual IDictionary<string, object> ActionArguments { get; }
    public virtual object Controller { get; }
    public virtual IActionResult Result { get; set; }
}

ActionExecutingContext繼承自FilterContext,我們暫且不關(guān)注它的父類,只看一下它自己的屬性。

  • ActionArguments 表示Action的參數(shù)列表,這里面放著各種從用戶接到請求參數(shù)以及其他中間處理程序添加的參數(shù)
  • Controller 表示執(zhí)行該請求的控制器,在之前我們提過,asp.net core 對于控制器的限制很小,所以控制器什么類型都可能,故而這里使用object作為控制器類型
  • Result 執(zhí)行結(jié)果,正常情況下,在此處獲取這個屬性的值沒有意義。但是我們可以通過修改這個屬性的值,來讓我們攔截請求

1.2.2 ActionExecutedContext

ActionExecutedContext 表示Action執(zhí)行完成后的上下文,這時候Action已經(jīng)執(zhí)行完成,我們可以通過這個獲取Action執(zhí)行結(jié)果:

public class ActionExecutedContext : FilterContext
{
    public virtual bool Canceled { get; set; }
    public virtual object Controller { get; }
    public virtual Exception Exception { get; set; }
    public virtual ExceptionDispatchInfo ExceptionDispatchInfo { get; set; }
    public virtual bool ExceptionHandled { get; set; }
    public virtual IActionResult Result { get; set; }
}

同樣,繼承自FilterContext,暫且忽略。

  • Canceled 表示是否被設(shè)置短路
  • Controller 處理請求的控制器
  • Exception 執(zhí)行過程中是否發(fā)生異常,如果有異常則 有值,否則為Null
  • ExceptionHandled 異常是否被處理
  • Result 此處對Result進行修改不會屏蔽執(zhí)行的ActionResult,但是可以向用戶隱藏對應(yīng)的實現(xiàn)

1.2.3 ResultExecutingContext

這是在Result渲染之前執(zhí)行的上下文,這時候Action已經(jīng)執(zhí)行完畢,正準(zhǔn)備渲染Result:

public class ResultExecutingContext : FilterContext
{
    public virtual bool Cancel { get; set; }
    public virtual object Controller { get; }
    public virtual IActionResult Result { get; set; }
}
  • Cancel 取消當(dāng)前結(jié)果執(zhí)行以及后續(xù)篩選器的執(zhí)行
  • Controller 控制器
  • Result 處理結(jié)果

1.2.4 ResultExecutedContext

Result已經(jīng)執(zhí)行完成了,獲取執(zhí)行結(jié)果上下文:

public class ResultExecutedContext : FilterContext
{
    public virtual bool Canceled { get; set; }
    public virtual object Controller { get; }
    public virtual Exception Exception { get; set; }
    public virtual ExceptionDispatchInfo ExceptionDispatchInfo { get; set; }
    public virtual bool ExceptionHandled { get; set; }
    public virtual IActionResult Result { get; }
}

這個類與 ActionExecutedContext類似,就不做介紹了。

1.2.5 FilterContext

在上面的四個上下文都繼承自 FilterContext,那么我們來看一下FilterContext中有哪些屬性或者方法:

public abstract class FilterContext : ActionContext
{
    public virtual IList<IFilterMetadata> Filters { get; }
    public TMetadata FindEffectivePolicy<TMetadata>() where TMetadata : IFilterMetadata;
}

可以看到FilterContext繼承了另一個ActionContext的類。小伙伴們應(yīng)該對這個類要有一定的概念,這個類是Action的上下文類。它完整存在于一個Action的生命周期,所以有時候可以通過ActionContext進行Action級的數(shù)據(jù)傳遞(不推薦)。

那么,繼續(xù)讓我們回過頭來看看ActionContext里有什么:

public class ActionContext
{
    public ActionDescriptor ActionDescriptor { get; set; }
    public HttpContext HttpContext { get; set; }
    public ModelStateDictionary ModelState { get; }
    public RouteData RouteData { get; set; }
}
  • ActionDescriptor 執(zhí)行的Action描述信息,包括Action的顯示名稱、一些參數(shù)等,具體用到的時候,再為大伙詳細說
  • HttpContext 可以通過這個屬性獲取此次請求的Request和Response對象
  • ModelState 模型校驗信息, 這部分在后續(xù)再為小伙伴們細說
  • RouteData 路由信息,asp.net core 在處理請求時解析出來的路由信息,包括在程序中修改的路由信息

2. 使用ActionFilter

在《【asp.net core 系列】9 實戰(zhàn)之 UnitOfWork以及自定義代碼生成》也就是上一篇中,介紹到了ActionFilter與普通特性類一致,可以通過標(biāo)注控制器然后啟用該ActionFilter。

因為大多數(shù)情況下,一個ActionFilter并不會僅僅局限于一個控制器,而是應(yīng)用于多個控制器。所以這時候,我們通常會設(shè)置一個基礎(chǔ)控制器,在這個控制器上進行標(biāo)注,然后讓子類繼承這個控制器。通過這種方式來實現(xiàn)一次聲明多次使用。

當(dāng)然,在asp.net core 中添加了另外的一種使用ActionFilter的方式,Setup.cs中

public void ConfigureServices(IServiceCollection services)
{
    services.AddControllersWithViews();
}

默認是這樣的,我們可以通過設(shè)置參數(shù)來添加一個全局應(yīng)用的Filter,例如說我們上一篇中創(chuàng)建的 UnitOfWorkFilterAttribute:

services.AddControllersWithViews(options=>
{
    options.Filters.Add<UnitOfWorkFilterAttribute>();
});

通過這種方式可以啟用一個全局ActionFilter。如果需要使用asp.net core的默認依賴注入可以使用 AddService進行配置。(依賴注入的內(nèi)容在后續(xù)會講解)。

3. 工具類生成

繼續(xù)上一篇遺留的內(nèi)容:

public static void CreateEntityTypeConfig(Type type)
{
    var targetNamespace = type.Namespace.Replace("Data.Models", "");
    if (targetNamespace.StartsWith("."))
    {
        targetNamespace = targetNamespace.Remove(0);
    }
    var targetDir = Path.Combine(new[] { CurrentDirect, "Domain.Implements", "EntityConfigures" }.Concat(
        targetNamespace.Split('.')).ToArray());

    if (!Directory.Exists(targetDir))
    {
        Directory.CreateDirectory(targetDir);
    }
    var baseName = type.Name.Replace("Entity", "");
    if (!string.IsNullOrEmpty(targetNamespace))
    {
        targetNamespace = $".{targetNamespace}";
    }

    var file = $"using {type.Namespace};" +
        $"\r\nusing Microsoft.EntityFrameworkCore;" +
        $"\r\nusing Microsoft.EntityFrameworkCore.Metadata.Builders;" +
        $"\r\nnamespace Domain.Implements.EntityConfigures{targetNamespace}" +
        "\r\n{" +
        $"\r\n\tpublic class {baseName}Config : IEntityTypeConfiguration<{type.Name}>" +
        "\r\n\t{" +
        "\r\n\t\tpublic void Configure(EntityTypeBuilder<SysUser> builder)" +
        "\r\n\t\t{" +
        $"\r\n\t\t\tbuilder.ToTable(\"{baseName}\");" +
        $"\r\n\t\t\tbuilder.HasKey(p => p.Id);" +
        "\r\n\t\t}\r\n\t}\r\n}";
    File.WriteAllText(Path.Combine(targetDir, $"{baseName}Config.cs"), file);
}

工具類其實本質(zhì)上就是一次文件寫入的方法,本身沒什么難度。

不過,這里還有有個小問題,每次調(diào)用都會覆蓋原有的文件,還有就是這里面有很多可以優(yōu)化的地方,小伙伴們可以自己試試去優(yōu)化一下,讓代碼更好看一點。

4 總結(jié)

到目前為止,實戰(zhàn)系列也有了幾篇,很多小伙伴問我能提供一下源碼嗎?當(dāng)然,能呀。不過不是現(xiàn)在,容我留個謎底。當(dāng)主要框架功能完成之后,我就會給小伙伴們發(fā)代碼的。

其實也是因為現(xiàn)在還沒個完整的,開放給小伙伴們也沒啥意義。當(dāng)然了,跟著一塊敲,也是能實現(xiàn)的哈。關(guān)鍵地方的代碼都有。

更多內(nèi)容煩請關(guān)注我的博客《高先生小屋》

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多