RazorEngine

目标

使用razorengine编译cshtml页面生静态html

制作一个vs2017的插件,实现右击cshtml文件时,编译该文件.

环境

razorengine4.5 / netframework4.6 / vs2017插件项目

razorengine参考文档  https://antaris.github.io/RazorEngine/index.html

测试实现

思路很简单,右击cshtml文件后,调用razorengine引擎,执行run方法得到编译后的html.

首先在控制台程序中测试了主要方法,发现没有问题.很快就编译出html了.编写cshtml页的语法部分支持razor,可以使用layout 程序语句 加载分部页等等.但在移到插件项目后,发现各种问题.

最大的问题在于速度很慢.简直无法使用.

以下是测试cshtml页面 使用了母板页,分部页,程序代码,raw函数,ViewBag.xxx,等常用的razor能力

// 主cshtml页

@{
layout="/layout.cshtml";
  ViewBag.Title="test-razorengine";
}
<div>@Raw("<h1>test razorengine</h1>")</div>
<table>
@for(int i=0;i<;i++){
<tr><td>@i</td><td>@i</td></tr>
}
</table>
@Include("part1.cshtml")

// razorengine编译使用

var config = new TemplateServiceConfiguration();
config.CachingProvider = new DefaultCachingProvider(t => { });
config.Language = Language.CSharp;
config.AllowMissingPropertiesOnDynamic = true;
config.DisableTempFileLocking = true;
using (Engine.Razor = RazorEngineService.Create(config))
{
// 添加这个cshtml页面以及它引用的母版页和片段页
foreach (string item in alltemps.Keys)
{
service.AddTemplate(item, alltemps[item]);
}
// 编译并运行得到编译后的静态html
string html =Engine.Razor.RunCompile("index.cshtml");
File.WriteAllText("index.html", html);
}

问题

1.页面缓存

测试时发现compile方法会编译cshtml然后缓存起来,即使修改了这个cshtml文件,重新添加到模板,依然不行,还是会使用之前编译过的缓存

此时需要重新添加模板,并且再编译之.在添加cshtml模板页到引擎时,每个页面的key是固定的,所以,需要先删除这个key再添加.否则会报key重复

// 删除KEY

((DelegateTemplateManager)config.TemplateManager).RemoveDynamic(k);

// 重新添加,依然是这个KEY,可以使用文件路径做KEY.

Engine.Razor.AddTemplate(k, alltemps[key]);

// 一定要编译,否则会使用缓存,即使重新添加了模板也不行

Engine.Razor.Compile(k);

2.编译卡死问题

在控制台中测试时并没有速度问题,即使每次重新编译都会很快,using()的作用是每次使用完引擎后,就释放资源,故每次都是重新初始化引擎,重新编译模板.

即使如此,依然很快.但是将代码移入到插件后,发现编译过程挂掉了.VS一直在运行,razorengine.compile这个方法卡住了,过了十分钟才结束运行...原因不知

这个问题的解决方法是使用roslyn这个新的编译器来编译.

config.CompilerServiceFactory = new RazorEngine.Roslyn.RoslynCompilerServiceFactory();
config.CompilerServiceFactory.CreateCompilerService(Language.CSharp);

3.编译过慢

Engine.Razor.Compile(k); 这个过程最慢,需要几百毫秒,但如果这个Key的页面是编译过的(缓存了),那么很快.

编译一个cshtml页面的相关页面比如母板页或者片段页,可能并不会一起修改.所以可以在每次重新加载模板前,判断文件是否修改过.

修改过的文件才重新添加编译,否则不用编译.如此,只耗废时间在那个修改过的文件上.

通过比较文件的MD5值来决定是否重新编译

优缺点

razor是个不错的工具,不仅可以用于写html,像代码生成器之类的都可以用它.确实是C#程序员的利器.比起T4模板来,更加友好一些,嵌套模板时也很容易理解.

razorengine这个编译工具有点慢,对于不会修改的模板存只存在首次慢的问题,但对于频繁修改模板那就麻烦了,每次都要重新编译,速度不行.另外,引擎缓存问题如果处置不当容易造成巨大内存占用.

上一篇:"类名.this"与"this"的区别


下一篇:RANSAC