由于两个ACM题花了差不多两个星期的时间,拖延了代码生成器的发布(下载)。
不过通过这段时间的折腾,对于C#细节优化有了更多的了解。知道了结构体数组元素赋值写个Set函数比=new要好,知道了静态字段的访问和引用成员字段访问一样低效(相对于堆栈访问),知道了临时变量的数量必须要自己控制(C#编译器的寄存器优化并不智能),知道了小函数内联是靠不住的,更加的重视非安全数组元素的访问次数了...
还是回到正题,上个月准备讨论一下代码生成器重新实现的问题,没有什么反馈,所以还是按照我以前的方案重写代码。我简单的说一下方案实现的原理与过程。
1、在需要代码生成功能的项目属性的后期生成事件中执行命令调用带参的fastCSharp.exe,代码生成器是基于.net元数据的,所以需要程序集相关信息:
if exist D:\fastCSharp\bin\Release\fastCSharp.exe D:\fastCSharp\bin\Release\fastCSharp.exe $(ProjectName) $(ProjectDir) $(TargetPath) $(TargetName)
ProjectName 是项目名称,预留字段
ProjectDir 是项目路径,用于查找或读写.cs程序文件
TargetPath 是程序集完整文件名,用于读取.NET元数据信息
TargetName 默认命名空间,用于生成目标文件命名,比如TargetName=fastCSharp,那么生成的代码文件名为fastCSharp.cSharper.cs
2、fastCSharp.exe找到所有继承自fastCSharp.setup.IAuto的类,读取其自定义属性配置fastCSharp.setup.auto,按照依赖关系fastCSharp.setup.auto.DependType拓扑排序,然后依次使用反射调用fastCSharp.setup.IAuto.Run。
list<keyValue<Type, setup.auto>> autos = setup.ui.CurrentAssembly.GetTypes()
.getFind(type => !type.IsInterface && !type.IsAbstract && type.isInterface(typeof(setup.IAuto)))
.getArray(type => new keyValue<Type, setup.auto>(type, type.customAttribute<setup.auto>(false)))
.getFind(value => value.Value != null && value.Value.IsSetup && value.Value.IsAuto);
setup.ui.Setup(autos, parameter, false);
当前版本的fastCSharp.exe有4个继承自fastCSharp.setup.IAuto的类:
fastCSharp.setup.cSharp.coder.cSharp是C#代码生成器的核心,它读取并解析所有程序模板并生成中间代码fastCSharp.cSharper.cs,fastCSharp.cSharper.cs就是一个字符串拼接程序。
string[] codes = parameter.Types.getArray(type => new definition { Type = type, Auto = type.customAttribute<auto>(false) })
.getFind(type => type.Auto != null && type.Auto.IsSetup && type.Auto.DependType == typeof(cSharper))
.getArray(type => type.ToString());
fastCSharp.setup.cSharp.ajax.cSharp是一个JSON处理代码生成实例的数据视图,其配套的程序模板是fastCSharp.setup.cSharp.template.ajax,自定义属性配置是fastCSharp.setup.cSharp.ajax。
fastCSharp.setup.cSharp.simpleTemplate是一个简单的自定义模板解析程序(与.NET元数据无关),比如template目录下的程序模板。
fastCSharp.setup.test用于简单的编译时单元测试。
3、代码生成实例的运行,比如fastCSharp.setup.cSharp.ajax.cSharp,它继承自fastCSharp.setup.IAuto,所以会被fastCSharp.exe默认调用。它会读取程序集TargetPath的元数据信息,获取所有标记自定义属性[fastCSharp.setup.cSharp.ajax]的类,读取其字段与属性信息生成最终的目标程序。
keyValue<Type, ajax>[] ajaxs = parameter.Types