Razor Engine,实现代码生成器的又一件利器

简介:

Razor Engine,之前仅仅是ASP.NET MVC的一种View引擎,目前已经完全成为一种可以独立使用的模版引擎,并且已经成为了CodePlex上一个开源的项目(http://razorengine.codeplex.com/)。对于使用过ASP.NET MVC Razor视图引擎的朋友们一定已经领略过它的灵活性和易用性,在这篇文章中我们将利用它来实现一个代码生成器使我们可以以Razor的语法来定义代码模版。[源代码从这里下载]

在《一个简易版的T4代码生成"框架"》这篇文章中,我创建了一个能够生成多个文件的代码生成器,其核心是两个基本的类型:Template类生成代码文件的内容,Generator执行Template。现在我将通过自定义一个基于Razor Engine的Template,让它读取以Razor语法编写的模版文件并生成相应的代码。

不过在具体介绍实现原理之前,还是先来看看它能够达到的效果。如右图所示,我们还是通过一个.tt文件(Generator.tt)来生成相应的代码文件,具体生成的代码文件就是依赖于该.tt文件的三个.cs 文件:Foo.cs、Bar.cs和Baz.cs。简单起见,我们仅仅在这三个文件中生成了一个与文件同名的空Class,它们的定义如下。

   1: public class Foo
   2: {
   3: }
   4:  
   5: public class Bar
   6: {
   7: }
   8:  
   9: public class Baz
  10: {
  11: }

如下所示的是Generator.tt的内容,它直接执行一个类型为DemoGenerator的生成器。

   1: <#@ template hostspecific="true" language="C#" #>
   2: <#@ assembly name="$(TargetDir)Artech.CodeGeneration.dll" #>
   3: <#@ import namespace="Artech.CodeGeneration" #>
   4: <#@ output extension=".empty" #>
   5: <#this.RunCodeGenerator(this.Host, new DemoGenerator());#>

作为代码生成器的DemoGenerator定义如下,它重写了CreateTemplates方法并返回一个包含三个RazorTemplate<TModel>的字典,该字典的Key表示生成的文件名。

   1: public class DemoGenerator: Generator
   2: {
   3:     protected override IDictionary<string, Template> CreateTemplates()
   4:     {
   5:         Dictionary<string, Template> templates = new Dictionary<string, Template>();
   6:         templates.Add("Foo.cs", new RazorTemplate<string>("template.razor", "Foo"));
   7:         templates.Add("Bar.cs", new RazorTemplate<string>("template.razor", "Bar"));
   8:         templates.Add("Baz.cs", new RazorTemplate<string>("template.razor", "Baz"));
   9:         return templates;
  10:     }
  11: }

构建RazorTemplate<TModel>需要传入两个参数,第一个参数是用于定义Razor模版的文件路径(可以是绝对路径,也可以是相对于.tt文件的相对路径),另一个是作为Model的对象(相当于ASP.NET强类型View的Model)。通过上面的代码可以看出,三个RazorTemplate<TModel>均使用同一个Model类型为字符串的模版文件template.razor,该模版内容定义如下。可以看出作为Model的字符串在模版中作为了生成类型的名称。

   1: public class 
   2: {
   3: }

我假设你已经了解了《一个简易版的T4代码生成"框架"》这篇文章的内容,所以我们只需要简单介绍RazorTemplate<TModel>的定义就可以了。首选RazorTemplate<TModel>是一个具有如下定义的RazorTemplate类型的子类。RazorTemplate的只读属性TemplateFileName 表示Razor模版的路径,用于读取模版文本内容的虚方法GetTemplateText直接读取指定模版文件的内容。在重写的TransformText方法中,我们调用Razor的静态方法Parse对模版内容进行解析并生成代码文本。

   1: public class RazorTemplate : Template
   2: {
   3:     public string TemplateFileName { get; private set; }
   4:     public RazorTemplate(string templateFileName)
   5:     {
   6:         this.TemplateFileName = templateFileName;
   7:     }
   8:     protected virtual string GetTemplateText()
   9:     {
  10:         string path = this.TemplateFileName;
  11:         if (!Path.IsPathRooted(path))
  12:         {
  13:             path = Path.Combine(Path.GetDirectoryName(TransformContext.Current.Host.TemplateFile), path);
  14:         }
  15:         return File.ReadAllText(path);
  16:     }
  17:  
  18:     public override string TransformText()
  19:     {
  20:         return Razor.Parse(this.GetTemplateText());
  21:     }
  22: }

继承自RazorTemplate的泛型RazorTemplate<TModel>定义如下。它具有一个表示Model的只读属性,该属性在构造函数中被初始化。在重写的TransformText方法中,我们调用Razor泛型的静态方法Parse<TModel>解析模版文本并传入Model对象。

   1: public class RazorTemplate<TModel> : RazorTemplate
   2: {
   3:     public TModel Model { get; private set; }
   4:     public RazorTemplate(string templateFileName, TModel model)
   5:         : base(templateFileName)
   6:     {
   7:         this.Model = model;
   8:     }
   9:  
  10:     public override string TransformText()
  11:     {
  12:         return Razor.Parse<TModel>(this.GetTemplateText(), this.Model);
  13:     }
  14: }

 

相关阅读:
与VS集成的若干种代码生成解决方案[博文汇总(共8篇)]
通过CodeDOM定义生成代码的结构
通过Visual Studio的Custom Tool定义代码生成器
不同于CodeDOM的代码生成机制——T4
通过T4模板实现单文件的代码生成
通过T4模板实现多文件的代码生成
解决T4模板的程序集引用的五种方案
编写T4模板进行代码生成无法避免的两个话题:"Assembly Locking"&"Debug"
通过自定义BuildProvider为ASP.NET提供代码生成


作者:蒋金楠
微信公众账号:大内老A
微博: www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号 蒋金楠的自媒体将会停用)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
相关文章
|
6月前
|
SQL 前端开发 Java
JSP软件产品管理系统myeclipse开发sql计算机程序web结构java编程网页源码
JSP软件产品管理系统是一套完善的毕业设计系统(servlet+dao+bean模式开发)MVC结构,对理解JSP java编程开发语言有帮助,系统具有完整的源代码和数据库,系统主要采用B/S模式开发
30 0
|
7月前
|
前端开发 关系型数据库 开发工具
构建自己的MVC框架(Ruby语言实现)-- 第一章 从零到“它工作了!”
构建自己的MVC框架(Ruby语言实现)-- 第一章 从零到“它工作了!”
|
10月前
|
监控 安全 测试技术
用 Python 撸一个 Web 服务器-第7章:重构——更好的组织代码
用 Python 撸一个 Web 服务器-第7章:重构——更好的组织代码
|
JSON JavaScript 前端开发
程序员接活利器,dataTable组件带你快速开发,摆脱CRUD
程序员接活利器,dataTable组件带你快速开发,摆脱CRUD
224 0
程序员接活利器,dataTable组件带你快速开发,摆脱CRUD
|
数据可视化 Java API
GUI 的 JavaWeb 鸡肋代码生成器 确定不了解了解?
GUI 的 JavaWeb 鸡肋代码生成器 确定不了解了解?
187 0
GUI 的 JavaWeb 鸡肋代码生成器 确定不了解了解?
|
开发框架 JSON 移动开发
基于.NET平台常用的框架整理
自从学习.NET以来,优雅的编程风格,极度简单的可扩展性,足够强大开发工具,极小的学习曲线,让我对这个平台产生了浓厚的兴趣,在工作和学习中也积累了一些开源的组件,就目前想到的先整理于此,如果再想到,就继续补充这篇日志,日积月累,就能形成一个自己的组件经验库。
1007 0
|
开发框架 PHP 容器
codeigniter-3.1 PHP开发框架
环境需求 最低硬件配置:1核CPU,1G内存(1+1)提示:如果你的应用较多,而主机节点的硬件配置较低,建议在部署节点时开通虚拟虚拟内存; 生产环境建议使用2G或以上内存; 推荐安装系统:Ubuntu-16.
984 0