CodeFirst时使用T4模板
我们都知道T4模板用于生成相似代码。
在DBFirst和ModelFirst条件下我们很容易从.edmx下获取所有实体类和其名称,并且通过我们定义的模板和某些遍历工作为我们生成所需要的相似代码。
但是CodeFirst模式下,我们没有edmx,从哪获取所有类名呢?难道要我们一个个把Entity实体层类名写进数组里吗,十个百个千个?
解决过程:
1.使用反射获取所有类名,并将此段代码写到T4模板中。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | foreach (Type t in Assembly.Load( "Entity" ).GetTypes()) { //你对这些类信息的操作 //e.g: if (t.FullName == "Entity.baseEntity" ) { } else { string fullName = t.FullName; int length = fullName.Length; int index = fullName.LastIndexOf( '.' ); string classname = fullName.Substring(index + 1, length - index - 1); nameStr += (classname) + "," ; } } |
2.结果如上代码可以加载程序集并且获取所有类名,可以自行断点查看。当我们把这段代码放到T4模板中,报错信息却是不能加载程序集。(看见了运行时文本模板,还没尝试那个模板是否有效)
3.最后实在没办法了,我在应用中创建了一个名为HelpTool的控制台程序。讲上段代码在Main中改写如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 | static void Main( string [] args) { using (StreamWriter writer = new StreamWriter( "D:\\EntitiesNames.txt" )) { string classNameStr = "" ; foreach (Type t in Assembly.Load( "Entity" ).GetTypes()) { if (t.FullName == "Entity.BaseEntity" || t.FullName == "Entity.DbContextFactory" || t.FullName == "Entity.WeDbContext" ) { } else { string fullName = t.FullName; int length = fullName.Length; int index = fullName.LastIndexOf( '.' ); string classname = fullName.Substring(index + 1, length - index - 1); classNameStr += (classname) + "," ; } } writer.Write( "" ); writer.Write(classNameStr); //Console.WriteLine("WriteClassNameSuccessfully"); //Console.ReadKey(); } } |
4.很明确,我将它写到文本文件里,每次修改实体类,我都要运行一下我的控制台应用。
5.在T4模板中读取文本文件。例如:
得到所有实体类的名,存到数组中,并遍历生成相似代码。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 | <#@ template language= "C#" debug= "false" hostspecific= "true" #> <#@ include file= "EF.Utility.CS.ttinclude" #> <#@ output extension= ".cs" #> <# string [] className= new string []{ "" }; using (StreamReader reader = new StreamReader( "D:\\EntitiesNames.txt" )) { string nameStr = reader.ReadToEnd(); int index=nameStr.LastIndexOf( ',' ); nameStr=nameStr.Substring(0,index); className= nameStr.Split( ',' ); } #> using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using DalFactory; using Entity; using s2s.IDAL; namespace s2s.UnitOfWork { public partial class UnitWork { <# foreach ( var entity in className) { #> public I<#=entity#>Dal <#=entity#>Dal { get { return AbstractDalFactory.GetInstance( "<#=entity#>Dal" ) as I<#=entity#>Dal; } } <#}#> } } |
我知道这个方法很笨,谈不上好方案,在遇到问题,coder们总能提出各种各样的解决方案。
希望园子里的朋友们推荐好的方案,多多指教。
分类:
.Net基础架构
· 智能桌面机器人:用.NET IoT库控制舵机并多方法播放表情
· Linux glibc自带哈希表的用例及性能测试
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 手把手教你在本地部署DeepSeek R1,搭建web-ui ,建议收藏!
· 新年开篇:在本地部署DeepSeek大模型实现联网增强的AI应用
· 程序员常用高效实用工具推荐,办公效率提升利器!
· Janus Pro:DeepSeek 开源革新,多模态 AI 的未来
· 【译】WinForms:分析一下(我用 Visual Basic 写的)