今天碰到这样一个需求,写的C#库,有时候需要在.net 2.0下编译,有时候需要在.net 4.0下编译,这个库里使用了lambda表达式,使用了扩展方法,使用了几个 System.Core.dll 引入的Action类型。
为了在 .net 2.0 下能够编译成功,我写了一个文件 Patch.cs,定义了 System.Runtime.CompilerServices.ExtensionAttribute 类型,这样就可以在2.0下使用lambda表达式和扩展方法了,同时,添加了几个用到的System.Core.dll 引入的Action类型:
- namespace System.Runtime.CompilerServices
- {
- public class ExtensionAttribute : Attribute { }
- }
- namespace System
- {
- public delegate void Action();
- public delegate void Action<T0,T1>(T0 t0,T1 t1);
- }
然而,要在.net 4.0 下编译,因为类型已经存在,必须注释掉Patch.cs,很麻烦。于是想通过条件编译来解决,即:
- #if NET2
- namespace System.Runtime.CompilerServices
- {
- public class ExtensionAttribute : Attribute { }
- }
- namespace System
- {
- public delegate void Action();
- public delegate void Action<T0,T1>(T0 t0,T1 t1);
- }
- #endif
问题是,.net 里没有定义和.net版本有关的指示符。怎么办呢?自己动手,丰衣足食,使用Build Events在编译之前自动侦测出项目所使用的.net版本,定义出我们想要的指示符。
在 C#模板编程(2): 编写C#预处理器,让模板来的再自然一点 一文中,写了一个程序 Csmacro.exe 来实现C#下的模板机制,本文在Csmacro.exe 的基础上,增加侦测项目所引用的.net 版本的功能。
原理:查找项目目录下的 csproj 文件,解析它,找到节点TargetFrameworkVersion,判断.net版本,然后生成一个Csmacro_Template.cs文件,在里面 #define 版本指示符。例如,对 .Net 2.0 项目,生成的 Csmacro_Template.cs 文件内容为:
- #define NET2
修改后Csmacro的代码可在:https://github.com/xiaotie/GebCommon 上下载(目前只处理了 .net 2.0 和 4.0,如需要针对其它版本,可自行修改代码)。有了 Csmacro,一切就好办了。
第一步,把 Csmacro.exe 放在Path路径下
第二步,打开需要条件编译的项目,添加 Pre-build 事件:Csmacro.exe $(ProjectDir)
第三步,编辑源文件,如,Patch.cs 文件修改为:
- #region include "Csmacro_Template.cs"
- #endregion
- #if NET2
- namespace System.Runtime.CompilerServices
- {
- public class ExtensionAttribute : Attribute { }
- }
- namespace System
- {
- public delegate void Action();
- public delegate void Action<T0,T1>(T0 t0,T1 t1);
- }
- #endif
#region include 是我引入的 Csmacro 宏语法。详见 C#模板编程(2): 编写C#预处理器,让模板来的再