上篇我們說到。編寫控制器類的步驟可總結為兩個:實現一個類,然後在該類中添加一些公有方法,在運行的該類的時候可作為控制器發現,而這些方法則作為操作被髮現。 這裡我們有兩個細節: 1:系統如何知道實例化那個控制器 2:如何確定用那個方法。 路由: 1:被傳統的路由發現,2:通過特性路由發現,3:通過混合 ...
上篇我們說到。編寫控制器類的步驟可總結為兩個:實現一個類,然後在該類中添加一些公有方法,在運行的該類的時候可作為控制器發現,而這些方法則作為操作被髮現。
這裡我們有兩個細節:
1:系統如何知道實例化那個控制器
2:如何確定用那個方法。
路由:
1:被傳統的路由發現,2:通過特性路由發現,3:通過混合路由策略發現,
傳統路由不做過多解釋。特性路由,可以讓URL模版與處理請求時使用的控制器和操作保持獨立,以後,即使URL進行修改,也不需要重構代碼。
混合路由則時前兩者一起使用,不過註意的是,特性定義的路由比傳統路由的優先順序更高。
POCO(plain Old C# Object)
控制器類可以是一個普通的傳統C#對象。如果想被髮現,要麼類名帶有Controller尾碼,要麼用Controller特性修飾該類。POCO簡單來說,它能減少開銷和/記憶體占用量。
訪問HTTP上下文
POCO最大的問題是沒有HTTP上下文,那麼我們可以通過ActionContext來實現如:
public class PocoController { [ActionController] public ActionContext Context{get;set} ...... }
操作篩選器
1:它是圍繞做方法運行的一段代碼,可用於修改和擴展方法本身的行為。
public interface IActionFilter { void OnActionExecuting(ActionExecutingContext filterContext); void OnActionExected(ActionExecutedContext filterContext); }
它提供了掛鉤,在操作之前和之後運行代碼。在篩選器內能夠訪問請求和控制器上下文,並且可以讀取和修改參數。
每個繼承了Cobtroller類的,用戶定義的控制器都會獲得IActionFilter介面的預設實現。,事實上,基類Controller提供了一對可重寫的方法,OnActionExecuting和OnActionExecuted。這就代表每個控制器類都有一個機會,用來決定在調用給定方法前,後或者調用方法前後做些什麼,只需要重寫基類的方法就能實現這種功能。當然POCO不具備.
protected DateTime StartTime; public override void OnActionExecuting(ActionExecutingContext context) { var action = context.ActionDescriptor.RouteValues["Action"]; if(string.Equals(action,"index",StringComparison.CurrentCultureIgnoreCase)) { StartTime = DateTime.Now; } base.OnActionExecuting(context); } public override void OnActionExecuted(ActionExecutedContext context) { var action = context.ActionDescriptor.RouteValues["Action"]; if (string.Equals(action, "index", StringComparison.CurrentCultureIgnoreCase)) { var timeSpan = DateTime.Now-StartTime; context.HttpContext.Response.Headers.Add( "duration", timeSpan.TotalMilliseconds.ToString()); } base.OnActionExecuted(context); }
計算執行了多少毫秒
2:篩選器的分類
:操作篩選器只是ASP.NET CORE 管道中調用的一種篩選器,按照篩選器實際完成的額任務,可分成不同的類型。
類型 | 描述 |
授權篩選器 | 管道中運行的第一個類篩選器,用來確定發出請求的用戶是否有權發出當前的請求 |
資源篩選器 | 當授權之後,在管道的其餘部分之前以及管道組件之後運行,對於緩存很有用 |
操作篩選器 | 在控制器方法操作之前和之後運行 |
異常篩選器 | 如果註冊,則在發生未處理異常時觸發 |
結果篩選器 | 在操作方法結果之前和之後運行 |
可以將篩選器應用單獨方法,也可以應用到整個控制器類,影響該控制器公開的所有操作方法,相對的,在應用程式啟動時註冊了全局篩選器之後,他們將自動應用到任何控制器類的任何操作。
①:添加自定義頭
public class HeaderAttribute:ActionFilterAttribute { public string Name { get; set; } public string Value { get; set; } public override void OnActionExecuted(ActionExecutedContext context) { if(!string.IsNullOrWhiteSpace(Name)&&!string.IsNullOrWhiteSpace(Value)) { context.HttpContext.Response.Headers.Add(Name, Value); } return; } }
② 設置請求的區域性
[AttributeUsage(AttributeTargets.Class|AttributeTargets.Method,AllowMultiple =false)] public class CultureAttribute:ActionFilterAttribute { public string Name { get; set; } public static string CookieName { get { return "_Culture"; } } public override void OnActionExecuting(ActionExecutingContext context) { var culture = Name; if (string.IsNullOrWhiteSpace(culture)) culture = GetSavedCultureOrDefault(context.HttpContext.Request); SetCultureOnThread(culture); base.OnActionExecuting(context); } private static void SetCultureOnThread(string language) { var cultureInfo = new CultureInfo(language); CultureInfo.CurrentCulture = cultureInfo; CultureInfo.CurrentUICulture = cultureInfo; } private static string GetSavedCultureOrDefault(HttpRequest request) { var culture = CultureInfo.CurrentCulture.Name; var cookie = request.Cookies[CookieName] ?? culture; return culture; } }
主要是在操作方法之前檢查一個名為_Culture的自定義cookie,其中包含了用戶首選的語言,如果沒找到cookie,篩選器預設使用當前區域性,並賦值給當前的線程。最後全局註冊
③:將方法限制只能Ajax調用
public class AjaxOnlyAttribute:ActionMethodSelectorAttribute { public override bool IsValidForRequest(RouteContext routeContext, ActionDescriptor action) => routeContext.HttpContext.Request.IsAjaxRequest(); }
public static class HttpRequestExtensions { public static bool IsAjaxRequest(this HttpRequest httpRequest) { if (httpRequest == null) throw new ArgumentException("request"); if (httpRequest.Headers != null) return httpRequest.Headers["X-Requested-With"] == "XMLHttpRequest"; return false; } }