asp.net core mvc實現偽靜態功能

來源:互聯網
上載者:User
這篇文章主要為大家詳細介紹了asp.net core mvc實現偽靜態功能的相關資料,具有一定的參考價值,感興趣的小夥伴們可以參考一下

在大型網站系統中,為了提高系統訪問效能,往往會把一些不經常變得內容發布成靜態頁,比如商城的產品詳情頁,新聞詳情頁,這些資訊一旦發布後,變化的頻率不會很高,如果還採用動態輸出的方式進行處理的話,肯定會給伺服器造成很大的資源浪費。但是我們又不能針對這些內容都獨立製作靜態頁,所以我們可以在系統中利用偽靜態方式進行處理,至於什麼是偽靜態,大家可以百度下。我們這裡就來介紹一下,在asp.net core mvc中實現偽靜態方式。

  mvc架構中,view代表的是視圖,它執行的結果就是最終輸出到用戶端瀏覽器的內容,包含html,css,js等。如果我們想實現靜態化,我們就需要把view執行的結果儲存成一個靜態檔案,儲存到指定的位置上,比如磁碟、分布式緩衝等,下次再訪問就可以直接讀取儲存的內容,而不用再執行一次商務邏輯。那asp.net core mvc要實現這樣的功能,應該怎麼做?答案是使用過濾器,在mvc架構中,提供了多種過濾器類型,這裡我們要使用的是動作過濾器,動作過濾器提供了兩個時間點:動作執行前,動作執行後。我們可以在動作執行前,先判斷是否已經產生了靜態頁,如果已經產生,直接讀取檔案內容輸出即可,後續的邏輯就執行跳過。如果沒有生產,就繼續往下走,在動作執行後這個階段捕獲結果,然後把結果產生的靜態內容進行儲存。

  那我們就來具體的實現代碼,首先我們定義一個過濾器類型,我們成為StaticFileHandlerFilterAttribute,這個類派生自架構中提供的ActionFilterAttribute,StaticFileHandlerFilterAttribute重寫基類提供的兩個方法:OnActionExecuted(動作執行後),OnActionExecuting(動作執行前),具體代碼如下:

[AttributeUsage(AttributeTargets.Class|AttributeTargets.Method, AllowMultiple = false, Inherited = false)]public class StaticFileHandlerFilterAttribute : ActionFilterAttribute{   public override void OnActionExecuted(ActionExecutedContext context){}   public override void OnActionExecuting(ActionExecutingContext context){}}

在OnActionExecuting中,需要判斷下靜態內容是否已經產生,如果已經產生直接輸出內容,邏輯實現如下:

//按照一定的規則產生靜態檔案的名稱,這裡是按照area+"-"+controller+"-"+action+key規則產生string controllerName = context.RouteData.Values["controller"].ToString().ToLower();string actionName = context.RouteData.Values["action"].ToString().ToLower();string area = context.RouteData.Values["area"].ToString().ToLower();//這裡的Key預設等於id,當然我們可以配置不同的Key名稱string id = context.RouteData.Values.ContainsKey(Key) ? context.RouteData.Values[Key].ToString() : "";if (string.IsNullOrEmpty(id) && context.HttpContext.Request.Query.ContainsKey(Key)){  id = context.HttpContext.Request.Query[Key];}string filePath = Path.Combine(AppContext.BaseDirectory, "wwwroot", area, controllerName + "-" + actionName + (string.IsNullOrEmpty(id) ? "" : ("-" + id)) + ".html");//判斷檔案是否存在if (File.Exists(filePath)){  //如果存在,直接讀取檔案  using (FileStream fs = File.Open(filePath, FileMode.Open))  {    using (StreamReader sr = new StreamReader(fs, Encoding.UTF8))    {       //通過contentresult返迴文件內容       ContentResult contentresult = new ContentResult();       contentresult.Content = sr.ReadToEnd();       contentresult.ContentType = "text/html";       context.Result = contentresult;    }  }}

  在OnActionExecuted中我們需要結果動作結果,判斷動作結果類型是否是一個ViewResult,如果是通過代碼執行這個結果,擷取結果輸出,按照上面一樣的規則,產生靜態頁,具體實現如下         

//擷取結果IActionResult actionResult = context.Result; //判斷結果是否是一個ViewResult    if (actionResult is ViewResult)    {      ViewResult viewResult = actionResult as ViewResult;      //下面的代碼就是執行這個ViewResult,並把結果的html內容放到一個StringBuiler對象中      var services = context.HttpContext.RequestServices;      var executor = services.GetRequiredService<ViewResultExecutor>();      var option = services.GetRequiredService<IOptions<MvcViewOptions>>();      var result = executor.FindView(context, viewResult);      result.EnsureSuccessful(originalLocations: null);      var view = result.View;      StringBuilder builder = new StringBuilder();       using (var writer = new StringWriter(builder))      {        var viewContext = new ViewContext(          context,          view,          viewResult.ViewData,          viewResult.TempData,          writer,          option.Value.HtmlHelperOptions);         view.RenderAsync(viewContext).GetAwaiter().GetResult();        //這句一定要調用,否則內容就會是空的        writer.Flush();      }      //按照規則產生靜態檔案名稱      string area = context.RouteData.Values["area"].ToString().ToLower();      string controllerName = context.RouteData.Values["controller"].ToString().ToLower();      string actionName = context.RouteData.Values["action"].ToString().ToLower();      string id = context.RouteData.Values.ContainsKey(Key) ? context.RouteData.Values[Key].ToString() : "";      if (string.IsNullOrEmpty(id) && context.HttpContext.Request.Query.ContainsKey(Key))      {        id = context.HttpContext.Request.Query[Key];      }      string devicedir = Path.Combine(AppContext.BaseDirectory, "wwwroot", area);      if (!Directory.Exists(devicedir))      {        Directory.CreateDirectory(devicedir);      }       //寫入檔案      string filePath = Path.Combine(AppContext.BaseDirectory, "wwwroot", area, controllerName + "-" + actionName + (string.IsNullOrEmpty(id) ? "" : ("-" + id)) + ".html");      using (FileStream fs = File.Open(filePath, FileMode.Create))      {        using (StreamWriter sw = new StreamWriter(fs, Encoding.UTF8))        {          sw.Write(builder.ToString());        }      }      //輸出當前的結果      ContentResult contentresult = new ContentResult();      contentresult.Content = builder.ToString();      contentresult.ContentType = "text/html";      context.Result = contentresult;    }

  上面提到的Key,我們直接增加對應的屬性

public string Key{  get;set;}

  這樣我們就可以使用這個過濾器了,使用的方法:在控制器或者控制器方法上增加 [StaticFileHandlerFilter]特性,如果想配置不同的Key,可以使用 [StaticFileHandlerFilter(Key="設定的值")]

  靜態化已經實現了,我們還需要考慮更新的事,如果後台把一篇文章更新了,我們得把靜態頁也更新下,方案有很多:一種是在後台進行內容更新時,同步把對應的靜態頁刪除即可。我們這裡介紹另外一種,定時更新,就是讓靜態頁有一定的有效期間,過了這個有效期間自動更新。要實現這個邏輯,我們需要在OnActionExecuting方法中擷取靜態頁的建立時間,然後跟目前時間對比,判斷是否已到期,如果未到期直接輸出內容,如果已到期,繼續執行後面的邏輯。具體代碼如下:

//擷取檔案資訊對象FileInfo fileInfo=new FileInfo(filePath);//結算時間間隔,如果小於等於兩分鐘,就直接輸出,當然這裡的規則可以改TimeSpan ts = DateTime.Now - fileInfo.CreationTime;if(ts.TotalMinutes<=2){  using (FileStream fs = File.Open(filePath, FileMode.Open))  {    using (StreamReader sr = new StreamReader(fs, Encoding.UTF8))    {      ContentResult contentresult = new ContentResult();      contentresult.Content = sr.ReadToEnd();      contentresult.ContentType = "text/html";      context.Result = contentresult;    }  }}

  到此偽靜態就實現好了。目前的處理方法,只能在一定程度上能夠提高訪問效能,但是針對大型的門戶系統來說,可能遠遠不夠。按照上面介紹的方式,可以再進行其他功能擴充,比如產生靜態頁後發行就緒到CDN上,也發行就緒到單獨的一個內容伺服器,等等。不管是什麼方式,實現思路都是一樣的。

以上就是本文的全部內容,希望對大家的學習有所協助,也希望大家多多支援topic.alibabacloud.com。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.