德语网站制作,深圳网站制作收费,企业网站建设的核心是,买外贸服装去哪个网站在大型网站系统中#xff0c;为了提高系统访问性能#xff0c;往往会把一些不经常变得内容发布成静态页#xff0c;比如商城的产品详情页#xff0c;新闻详情页#xff0c;这些信息一旦发布后#xff0c;变化的频率不会很高#xff0c;如果还采用动态输出的方式进行处理… 在大型网站系统中为了提高系统访问性能往往会把一些不经常变得内容发布成静态页比如商城的产品详情页新闻详情页这些信息一旦发布后变化的频率不会很高如果还采用动态输出的方式进行处理的话肯定会给服务器造成很大的资源浪费。但是我们又不能针对这些内容都独立制作静态页所以我们可以在系统中利用伪静态的方式进行处理至于什么是伪静态大家可以百度下。我们这里就来介绍一下在asp.net core mvc中实现伪静态的方式。 mvc框架中view代表的是视图它执行的结果就是最终输出到客户端浏览器的内容包含html,css,js等。如果我们想实现静态化我们就需要把view执行的结果保存成一个静态文件保存到指定的位置上比如磁盘、分布式缓存等下次再访问就可以直接读取保存的内容而不用再执行一次业务逻辑。那asp.net core mvc要实现这样的功能应该怎么做答案是使用过滤器在mvc框架中提供了多种过滤器类型这里我们要使用的是动作过滤器动作过滤器提供了两个时间点动作执行前动作执行后。我们可以在动作执行前先判断是否已经生成了静态页如果已经生成直接读取文件内容输出即可后续的逻辑就执行跳过。如果没有生产就继续往下走在动作执行后这个阶段捕获结果然后把结果生成的静态内容进行保存。 那我们就来具体的实现代码首先我们定义一个过滤器类型我们成为StaticFileHandlerFilterAttribute这个类派生自框架中提供的ActionFilterAttributeStaticFileHandlerFilterAttribute重写基类提供的两个方法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-actionkey规则生成 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.GetRequiredServiceViewResultExecutor(); var option services.GetRequiredServiceIOptionsMvcViewOptions(); 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.TotalMinutes2) { 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上也可以发布到单独的一个内容服务器等等。不管是什么方式实现思路都是一样的。