ASP.NET MVC 控制器

  上篇咱們說到。編寫控制器類的步驟可總結爲兩個:實現一個類,而後在該類中添加一些公有方法,在運行的該類的時候可做爲控制器發現,而這些方法則做爲操做被發現。緩存

這裏咱們有兩個細節: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;
        }
    }
相關文章
相關標籤/搜索