最近遇到一个项目,要使用RazorEngine做模板引擎,然后完成简易的CMS功能,以减轻重复的CDRU操作,同时复用管理后台。没错,使用的正是GIT HUB上的开源项目:https://github.com/Antaris/RazorEngine 。模板编译过程非常耗时,所以Razor提供了Compile和Parse的带key参数的重载,以实现从缓存中加载编译后的模板的功能。不过这里还是有一个问题,对于web项目而言,应用程序池会周期性的回收(即使设置了不自动回收,不知为何)。所以,仍然会存在重新编译而导致页面长时间挂起的问题。或许可以提供一个进度条,告诉客户这是一个高大上的东西,需要热身....不过应该有更好的办法,就是从RazorEngine本身着手。

RazorEngine接收到模板内容的时候,会调用编辑器将其编译成一个程序集,加载到内存中,同时返回编译好的对应的模板的类型对象(Type对象)。然后调用对象的构造函数,生成对象实例,最后“执行”模板。BUT!为什么是加载到内存呢,如果是将程序集保存在磁盘上,那么下次再进行读取的话,这个性能绝对不是同一个档次的。所以,考虑之后,决定用以下的思路解决问题:

获取到模板之后(字符串),会计算其MD5值,并作为生成的模板类型的Class(有坑),同时将其作为程序集的名称。每当客户机代码请求编译模板的时候,就去指定的目录查找是否有这么一个程序集,如果有,直接加载这个程序集,并返回类型信息(因为程序集中只有一个类型,所以非常方便)。RazorEngine本身采用了可扩展的设计。扩展口就在Razor.SetTemplateService()。只需要实现CompilerServiceBase抽象类,就能自定义模板引擎的逻辑。只是...代码编译部分处于调用的较底层,如果全部采用全新的实现的话...工作量不少,而且容易存在BUG。SO...原样应用RazorEngine.dll并进行扩展这么完美的事情暂时还办不到。我采用的做法是,修改RazorEngine的源码,但是不修改已经定义的类型,只在相应的名称空间下面提供新的类型来满足自己的需求(因为其中不少需要的类型是internal的)。

参考TemplateService的源码,实现了一个ReloadableTemplateService,重新实现了CreateTemplateType方法:

[Pure]
public virtual Type CreateTemplateType(string razorTemplate, Type modelType)
{
//重要:类名不能以数字开头
var key = GetTemplateMd5(razorTemplate);
string className = "C" + key;
var assemblyPath = AssemblyDirecotry.TrimEnd(new[] { '/', '\\' }) + "\\" + key + ".dll"; if (File.Exists(assemblyPath))
{
try
{
//var assembly = Assembly.Load(File.ReadAllBytes(assemblyPath));
var assembly = Assembly.LoadFile(assemblyPath);
return assembly.GetTypes()[];
}
catch (Exception ex)
{
Debug.WriteLine("an error ocured and assembly has been deleted");
File.Delete(assemblyPath);
}
} var context = new TypeContext
{
ModelType = (modelType == null) ? typeof(object) : modelType,
TemplateContent = razorTemplate,
TemplateType = (_config.BaseTemplateType) ?? typeof(TemplateBase<>),
}; foreach (string ns in _config.Namespaces)
context.Namespaces.Add(ns); //csharp only
var service = new CSharpFileDirectCompilerService(); service.Debug = _config.Debug;
service.CodeInspectors = _config.CodeInspectors ?? Enumerable.Empty<ICodeInspector>(); var result = service.CompileType(context, className, assemblyPath);
_assemblies.Add(result.Item2); return result.Item1;
}

需要注意是,C#的类型名称不能以数字开头...这个问题我查了一下午,主要是压根没有想到这一点。生成程序集的方法,被放置在CSharpFileDirectCompilerService中,由于这个类型的局限性很强(我只是想快速解决问题),所以没有实现基类要求的方法(我把它废了,虽然这样很傻逼):

[Pure]
private Tuple<CompilerResults, string> Compile(TypeContext context, string className, string assemblyPath)
{
if (_disposed)
throw new ObjectDisposedException(GetType().Name); var compileUnit = GetCodeCompileUnit(className, context.TemplateContent, context.Namespaces,
context.TemplateType, context.ModelType); var @params = new CompilerParameters
{
GenerateInMemory = false,
OutputAssembly = assemblyPath,
GenerateExecutable = false,
IncludeDebugInformation = false,
CompilerOptions = "/target:library /optimize /define:RAZORENGINE"
}; var assemblies = CompilerServicesUtility
.GetLoadedAssemblies()
.Where(a => !a.IsDynamic && File.Exists(a.Location))
.GroupBy(a => a.GetName().Name)
.Select(grp => grp.First(y => y.GetName().Version == grp.Max(x => x.GetName().Version)))
// only select distinct assemblies based on FullName to avoid loading duplicate assemblies
.Select(a => a.Location); var includeAssemblies = (IncludeAssemblies() ?? Enumerable.Empty<string>());
assemblies = assemblies.Concat(includeAssemblies)
.Where(a => !string.IsNullOrWhiteSpace(a))
.Distinct(StringComparer.InvariantCultureIgnoreCase); @params.ReferencedAssemblies.AddRange(assemblies.ToArray()); string sourceCode = null;
if (Debug)
{
var builder = new StringBuilder();
using (var writer = new StringWriter(builder, CultureInfo.InvariantCulture))
{
_codeDomProvider.GenerateCodeFromCompileUnit(compileUnit, writer, new CodeGeneratorOptions());
sourceCode = builder.ToString();
}
} return Tuple.Create(_codeDomProvider.CompileAssemblyFromDom(@params, compileUnit), sourceCode);
} /// <summary>
/// Compiles the type defined in the specified type context.
/// </summary>
/// <param name="context">The type context which defines the type to compile.</param>
/// <returns>The compiled type.</returns>
[Pure, SecurityCritical,Obsolete("该方法无法兼容其父类,功能已经在其重载中提供")]
public override Tuple<Type, Assembly> CompileType(TypeContext context)
{
throw new NotImplementedException("该方法无法兼容其父类,功能已经在其重载中提供");
} /// <summary>
///
/// </summary>
/// <param name="context"></param>
/// <param name="className"></param>
/// <param name="assemblyDirectory"></param>
/// <returns></returns>
/// <exception cref="NullReferenceException"></exception>
/// <exception cref="TemplateCompilationException"></exception>
public Tuple<Type, Assembly> CompileType(TypeContext context, string className, string assemblyDirectory)
{
if (context == null)
throw new NullReferenceException("context");
var result = Compile(context, className, assemblyDirectory);
var compileResult = result.Item1; if (compileResult.Errors != null && compileResult.Errors.HasErrors)
throw new TemplateCompilationException(compileResult.Errors, result.Item2, context.TemplateContent); return Tuple.Create(
compileResult.CompiledAssembly.GetType("CompiledRazorTemplates.Dynamic." + className),
compileResult.CompiledAssembly);
}

然后就成了,调用方式是:

            Razor.SetTemplateService(new ReloadableTemplateService()
{
AssemblyDirecotry = "d:\\temple"
});

稍微测了下性能,这里定性描述下:编译模板的时候,耗时会根据模板的大小和复杂度,一两秒或者更多。而加载一个程序集的话,尤其是像这种一个类型的小程序集,总是毫秒级的。
真是个愉快的周末。

DIY RazorEngine 的程序集生成方式的更多相关文章

  1. Razor - 模板引擎 / 代码生成 - RazorEngine

    目录 Brief Authors Official Website RazorEngine 的原理 - 官方解释 安装记录 Supported Syntax (默认实现支持的语法) 测试记录 - ca ...

  2. 从invoke简单理解反射

    前言 程序集   : 程序集是.NET应用程序的基本单位,包含了程序的资源.类型元数据和MSIL代码.根据程序集生成方式的不同,可分为静态程序集和动态程序集.程序集又可分为单文件程序集和多文件程序集, ...

  3. C#开发问题汇总

    问题1:HTTP 错误 500.21 - Internal Server Error处理程序“NickLeeCallbackHandler”在其模块列表中有一个错误模块“ManagedPipeline ...

  4. “RazorEngine.Templating.TemplateCompilationException”类型的异常在 RazorEngine.NET4.0.dll 中发生,但未在用户代码中进行处理

    错误信息: "RazorEngine.Templating.TemplateCompilationException"类型的异常在 RazorEngine.NET4.0.dll 中 ...

  5. 第四篇 基于.net搭建热插拔式web框架(RazorEngine实现)

    在开头也是先给大家道个歉,由于最近准备婚事导致这篇文章耽误了许久,同时也谢谢老婆大人对我的支持. 回顾上篇文章,我们重造了一个controller,这个controller中用到了视图引擎,我们的视图 ...

  6. 解决T4模板的程序集引用的五种方案

    在众多.NET应用下的代码生成方案中,比如CodeDOM,BuildProvider, 我觉得T4是最好的一种.关于T4的基本概念和模板结果,可以参考我的文章<基于T4的代码生成方式>.如 ...

  7. Razor模板引擎 (RazorEngine)

    Razor模板引擎不仅在ASP.NET MVC中内置了Razor模板引擎,还有一个开源的RazorEngine, 这样以来我们可以在非ASP.NET MVC项目中使用Razor引擎,甚至在控制台,Wi ...

  8. MVC的验证(模型注解和非侵入式脚本的结合使用) .Net中初探Redis .net通过代码发送邮件 Log4net (Log for .net) 使用GDI技术创建ASP.NET验证码 Razor模板引擎 (RazorEngine) .Net程序员应该掌握的正则表达式

    MVC的验证(模型注解和非侵入式脚本的结合使用)   @HtmlHrlper方式创建的标签,会自动生成一些属性,其中一些属性就是关于验证 如图示例: 模型注解 通过模型注解后,MVC的验证,包括前台客 ...

  9. [转]从数据到代码——基于T4的代码生成方式

    本文转自:http://www.cnblogs.com/artech/archive/2010/10/23/1859529.html 在之前写一篇文章<从数据到代码>(上篇.下篇)中,我通 ...

随机推荐

  1. 我的第二篇--nginx安装问题之路径问题

    这几天还是一直在搭建nginx,并且要在nginx的基础之上配置naxsi(WAF防火墙)并使它生效,但是随之而来的问题也会有很多,也许因为我是个新手,所以遇到的问题要多,不解的问题也要很多,不知道又 ...

  2. 带有可点击区域的图像映射:HTML <map> 标签

    实例 带有可点击区域的图像映射: <img src="planets.jpg" border="0" usemap="#planetmap&qu ...

  3. OQL对象查询语言

    在用mat工具分析内存使用情况查询OutOfMemory原因时,OQL会有很大帮助,所以先在这里总结一下. 基本语法: select <javascript expression to sele ...

  4. PullToRefresh的个性化扩展

    一:实现区别下拉刷新和上拉加载 参考资料:http://blog.csdn.net/losetowin/article/details/18261389 在PullToRefresh的类库的com.h ...

  5. 对css float 浮动的学习心得

    css float浮动详解 @(css float)[hasLayout|clear float|妙瞳] css float的定义和用法 float 属性定义元素在哪个方向浮动.以往这个属性总应用于图 ...

  6. webstorm添加vue模板支持

    字谕纪泽: 八月一日,刘曾撰来营,接尔第二号信并薛晓帆信,得悉家中四宅平定,至以为尉. 汝读”四书”无甚心得,由不能虚心涵泳,切己体察.朱子教人读书之法,此二语最为精当.尔现读”离娄”,即如“离娄”首 ...

  7. android手机震动

    Vibrator是安卓提供的震动器,其没有构造器,通过getSystemService(Context.VIBRATOR_SERVICE)方法获取对象.但使用此类时需要在清单文件中添加访问权限andr ...

  8. CefSharp开源库的使用(一)

    关于CEF: 嵌入式Chromium框架(简称CEF) 是一个由Marshall Greenblatt在2008建立的开源项目,它主要目的是开发一个基于Google Chromium的Webbrows ...

  9. oc 一些通用函数

    1 i= 0,1,2... unichar c = [self characterAtIndex:i]; //取出i这个位置对应的字符 2 拼凑字符串 [NSString stringWithForm ...

  10. hdu 1753 大明A+B(高精度小数加法)

    //深刻认识到自己的粗心,为此浪费了一天.. Problem Description 话说,经过了漫长的一个多月,小明已经成长了许多,所以他改了一个名字叫"大明". 这时他已经不是 ...