上篇咱們說到。編寫控制器類的步驟可總結爲兩個:實現一個類,而後在該類中添加一些公有方法,在運行的該類的時候可做爲控制器發現,而這些方法則做爲操做被發現。緩存
這裏咱們有兩個細節:cookie
1:系統如何知道實例化那個控制器ide
2:如何肯定用那個方法。this
路由:spa
1:被傳統的路由發現,2:經過特性路由發現,3:經過混合路由策略發現,線程
傳統路由不作過多解釋。特性路由,可讓URL模版與處理請求時使用的控制器和操做保持獨立,之後,即便URL進行修改,也不須要重構代碼。code
混合路由則時前二者一塊兒使用,不過注意的是,特性定義的路由比傳統路由的優先級更高。對象
POCO(plain Old C# Object)blog
控制器類能夠是一個普通的傳統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; } }