首页 > 编程 > .NET > 正文

asp.net模板引擎Razor中cacheName的问题分析

2024-07-10 12:47:54
字体:
来源:转载
供稿:网友

本文实例讲述了asp.net模板引擎Razor中cacheName的问题。。具体如下:

一、为什么使用cacheName

使用cacheName主要是考虑到Razor.Parse()每解析一次都会动态创建一个程序集,如果解析量很大,就会产生很多程序集,大量的程序集调用会造成程序非常慢。

举个例子:

如果编译1000次,编译速度就会很慢。

static void Main(string[] args){ string cshtml = File.ReadAllText(@"E:/百度云同步盘/Study/Net_ASP.NET/Web基本原理/RazorCacheNameTest/HTMLPage1.cshtml"); for (int i = 0; i < 1000; i++) {  string html = Razor.Parse(cshtml);  } Assembly[] asms = AppDomain.CurrentDomain.GetAssemblies(); foreach (Assembly asm in asms) {  Console.WriteLine(asm.FullName+"/r/n"); } Console.ReadKey();}

二、如何解决这个问题

使用Razor.Parse()时,带上cacheName参数。

指定一个cacheName叫cc,下次Parse()解析时就不会重新编译了(除非cshtml内容修改,那么cacheName名也要重新命名,让Parse()解析新文件)

for (int i = 0; i < 1000; i++){  //如果调用1000次,使用下面方式就会创建很多程序集,性能很低  string html = Razor.Parse(cshtml);   //解析的cshtml文件我给的一个“缓存名”是cc,这次一旦编译成功  //下次再让你Parse() cc就不用重复编译了,速度会非常快,  //除非cshtml内容修改  Razor.Parse(cshtml, null, "cc");}

三、怎么确定cacheName表示的文件已修改呢?

有两种方式,一种就是文件全路径+文件修改时间,还可以根据cshtml文件的MD5值。

for (int i = 0; i < 10; i++){  string cshtml = File.ReadAllText(fullPath);  string cacheName = fullPath + File.GetLastWriteTime(fullPath);  //文件全路径+文件上一次被修改时间  string html = Razor.Parse(cshtml,null,cacheName);  Console.WriteLine(html);  Console.ReadKey();}

每当cshtml文件被修改,cacheName的值就会改变,Parse()根据cacheName值判断是否重新编译。假如测试过程中对cshtml文件做了三次修改,最终会生成三个程序集,如果cshtml文件未修改,最后只有一个程序集。

注意:关于cacheName的问题。

经过试验发现,即使cacheName写成一个固定的值,当cshtml发生改变的时候Parse的结果也是修改后的内容,这是为什么呢?

经过反编译我们发现Parse方法最终调用的是TemplateService的GetTemplate方法,代码如下:

private ITemplate GetTemplate<T>(string razorTemplate, object model, string cacheName){ Func<string, CachedTemplateItem, CachedTemplateItem> updateValueFactory = null; CachedTemplateItem item; if (razorTemplate == null) {  throw new ArgumentNullException("razorTemplate"); } int hashCode = razorTemplate.GetHashCode(); if (!this._cache.TryGetValue(cacheName, out item) || (item.CachedHashCode != hashCode)) {  Type templateType = this.CreateTemplateType(razorTemplate, (model == null) ? typeof(T) : model.GetType());  item = new CachedTemplateItem(hashCode, templateType);  if (updateValueFactory == null)  {   updateValueFactory = (n, i) => item;  }  this._cache.AddOrUpdate(cacheName, item, updateValueFactory); } return this.CreateTemplate(null, item.TemplateType, model);}            
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表