CodeModel的范例:一个为项目所有相关类添加工厂方法的AddIn

1、问题描述

      对一个程序做性能优化,发现程序里会大量创建动态对象,是影响性能的一个瓶颈。程序里都是采用Activator.CreateInstance(Type)的方法,记得在codeproject看过一篇文章(原文在此:Dynamic Objects, Factories, and Runtime Machines to Boost Performance),对动态创建对象的几种方式进行效率对比,Activator.CreateInstance是效率较低的一种。采用FormatterServices.GetUninitializedObject得到一个未初始化的类,再调用类的工厂方法,效率有大幅提高。下面是简单的代码说明。更多内容请参见上文提到的文章。
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Runtime.Serialization;


  5. //第一种方式:Activator.CreateInstance(Type)方式。效率较低
  6. namespace ActivatorCreator
  7. {
  8.     public class Widget { }
  9.     public class WidgetA : Widget { }
  10.     public class WidgetB : Widget { }
  11.     //..
  12.     public class WidgetZ : Widget { }
  13.     
  14.     public abstract class Creator
  15.     {
  16.         public static Widget DynamicCreate(Widget w)
  17.         {
  18.             return (Widget)Activator.CreateInstance(w.GetType());
  19.         }
  20.     }
  21. }
  22. //第二种动态创建对象方式
  23. //采用FormatterServices.GetUninitializedObject(Type)
  24. //获得一个未初始化对象,再调用对象的工厂方法得到对象。
  25. //经测试时间效率大概是前一种的150倍左右。
  26. namespace SerializationCreator
  27. {
  28.     public class Widget
  29.     {
  30.         public virtual Widget GetInstance()
  31.         {
  32.             return new Widget();
  33.         }
  34.     }
  35.     public class WidgetA : Widget
  36.     {
  37.         public override Widget GetInstance()
  38.         {
  39.             return new WidgetA();
  40.         }
  41.     }
  42.     public class WidgetB : Widget
  43.     {
  44.         public override Widget GetInstance()
  45.         {
  46.             return new WidgetB();
  47.         }
  48.     }
  49.     //
  50.     public class WidgetZ : Widget
  51.     {
  52.         public override Widget GetInstance()
  53.         {
  54.             return new WidgetZ();
  55.         }
  56.     }

  57.     //Serialization配合工厂方法动态生成对象,高效率
  58.     public abstract class Creator
  59.     {
  60.         public static Widget DynamicCreate(Widget w)
  61.         {
  62.             Widget widgetFactory = 
  63.                 (Widget)FormatterServices.GetUninitializedObject(w.GetType());
  64.             return widgetFactory.GetInstance();
  65.         }
  66.     }
  67. }
复制代码
因此,决定采用FormatterServices.GetUninitializedObject方法来代替Activator.CreateInstance(Type)。整个项目有两个基类以及从该基类继承的大量子类(数目大概有100+)需要更改,而且继承关系最深达到5层。代码的修改工作量不小。

2、解决办法


2.1 复制粘贴

大概是最常规也最无趣的方法,我在复制了10个左右的类后实在受不了这种单调和枯燥,放弃了。


2.2 Code Snippet

复制的进阶,就是用Code Snippet了。把相同代码做成Snippet,效率有大幅提高。Snippet文件内容如下:
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <CodeSnippets xmlns="http://schemas.microsoft.com/VisualStudio/2005/CodeSnippet">
  3.   <CodeSnippet Format="1.0.0">
  4.     <Header>
  5.       <Title>Add Factory Method</Title>
  6.       <Author>lumber</Author>
  7.       <Description>为类添加工厂方法</Description>
  8.       <HelpUrl></HelpUrl>
  9.       <SnippetTypes />
  10.       <Keywords />
  11.       <Shortcut>fm</Shortcut>
  12.     </Header>
  13.     <Snippet>
  14.       <References />
  15.       <Imports />
  16.       <Declarations>
  17.         <Literal Editable="false">
  18.           <ID> RetType </ID>
  19.           <Type></Type>
  20.           <ToolTip>返回的类型</ToolTip>
  21.           <Default> RetType </Default>
  22.     <!--注意这里的Function,以及ClassName()函数-->
  23.           <Function>ClassName()</Function>
  24.         </Literal>
  25.       </Declarations>
  26.       <Code Language="csharp" Kind="" Delimiter="$">
  27.         <![CDATA[internal override CADEntityData GetInstance()
  28.       {
  29.           return new $RetType$();
  30.       }]]>
  31.       </Code>
  32.     </Snippet>x
  33.   </CodeSnippet>
  34. </CodeSnippets>
复制代码
在上面的snippet中,值得一提的是,使用了snippet function。即先定义了一个Literal,名称为RetType,代表工厂方法返回类型。我们知道不同的子类,工厂方法的函数签名相同,不同的是返回该类的实例。即RetType的值要等于被插入的类的名称。于是我们为RetType这个Literal提供了一个function,ClassName(),该函数返回snippet所在类的名称。

      实际我们不需要手工来写这个文件,这里推荐Code Snippet Editor这个小 工具


2.3 CodeModel

    现在只需要找到需要更改的类,敲下快捷键fm,再双击tab就ok了,委实比当初复制、编辑幸福多了。但人心难足,Snippet还是有不爽的地方:

      1、找到并打开所有要修改的类(100+啊兄弟,项目里每个类都是一个单独文件)不停地重复按键,也是挺无聊的活。更重要的是,要保证不能遗漏——上文说了,很多子类都经过了多达5层的继承……

      2、不同的基类要重新编制一个snippet。另外考虑到,如果以后别的项目要有类似的更改呢?如果项目是用VB.NET而不是c#呢?。。。。

    看来,最好的解决办法是写一个AddIn了。

      简单分析下任务,其实就是两个:1、寻找项目中要添加工厂方法的基类及所有派生类。2、为这些类添加一个相应的工厂函数。

      很自然需要用到CodeModel。

      关于VS的扩展开发,推荐园子里Anders Cui的 系列文章。而Anders Cui的系列恰好没有写关于CodeModel的内容,既有珠玉在前,所以小可也就斗胆续貂,简单写写关于CodeModel的内容。

    直接用Vs2008新建项目,选择“Visual Stduio外接程序”,选择使用c# 语言开发,然后一路默认,编辑器就自动为你生成了一个AddIn项目。该项目已经自动生成了大部分代码,包括将的AddIn程序添加到工具 菜单上等。接下来只需要将需要执行的代码加入到Exec函数中即可。

    我们再在项目中添加一个含文本框的窗体,作为输入界面,输入项目中要添加工厂方法的基类的全名(含命名空间)。接下来就是利用CodeModel寻找该类及其派生类,并添加工厂方法。还是代码说话吧,相关函数我在注释里都有说明,另外可以查询msdn。
  1. /// <summary>
  2. /// 实现 IDTCommandTarget 接口的 Exec 方法。此方法在调用该命令时调用。编辑器自动生成。
  3. /// </summary>
  4. /// <param term='commandName'>要执行的命令的名称。</param>
  5. /// <param term='executeOption'>描述该命令应如何运行。</param>
  6. /// <param term='varIn'>从调用方传递到命令处理程序的参数。</param>
  7. /// <param term='varOut'>从命令处理程序传递到调用方的参数。</param>
  8. /// <param term='handled'>通知调用方此命令是否已被处理。</param>
  9. /// <seealso class='Exec' />
  10. public void Exec(string commandName, vsCommandExecOption executeOption,
  11.     ref object varIn, ref object varOut, ref bool handled)
  12. {
  13.     handled = false;
  14.     if (executeOption == vsCommandExecOption.vsCommandExecOptionDoDefault)
  15.     {
  16.         if (commandName == "AddFactoryMethod.Connect.AddFactoryMethod")
  17.         {
  18.             InputBox ibox = new InputBox();
  19.             //程序添加一个输入窗口,输入要添加工厂方法的基类的全称(含命名空间)
  20.             if (ibox.ShowDialog() == DialogResult.OK 
  21.                       && !string.IsNullOrEmpty(ibox.Str))
  22.             {
  23.                 //获取当前解决方案的第一个项目的CodeModel
  24.                 //注意CodeModel中集合都是以下标1开始而非0。如Projects.Item(1)
  25.                 CodeModel cm = _applicationObject.Solution.Projects.Item(1).CodeModel;
  26.                 foreach (CodeElement ce in cm.CodeElements)
  27.                 {
  28.                     //只有命名空间和类中才包含类。
  29.                     if ((ce is CodeNamespace) | (ce is CodeClass))
  30.                     {

  31.                         SetAllClass(ce, ibox.Str);
  32.                     }
  33.                 }
  34.             }

  35.             handled = true;
  36.             return;
  37.         }
  38.     }
  39. }
  40. /// <summary>
  41. /// 寻找基类及所有子类,并添加工厂方法。
  42. /// 这个函数主要演示了:
  43. /// 1、如何寻找一个项目中包含的所有类。
  44. /// 注意,所有的类应该包含在项目的命名空间和类中。
  45. /// 2、如何判断类是指定类的派生类。
  46. /// 3、CodeElement(包括CodeClass、CodeFunction……)等一系列对象的基本用法。
  47. /// </summary>
  48. /// <param name="ct">类或者命名空间</param>
  49. /// <param name="str">基类的全称(含命名空间)</param>
  50. private void SetAllClass(CodeElement ct, string str)
  51. {
  52.     CodeClass cc = ct as CodeClass;
  53.     if (cc != null && cc.get_IsDerivedFrom(str))
  54.     {
  55.         //cc.get_IsDerivedFrom(str)函数用来判断类是否由全名为str的类派生得来。
  56.         //注意!自身也是自身的派生类,即cc.get_IsDerivedFrom(cc.FullName)将返回true。
  57.         //为基类和派生类添加工厂方法
  58.         AddFactoryMethod(cc, str);
  59.     }
  60.     CodeElements elements;
  61.     if (ct is CodeNamespace)
  62.     {

  63.         elements = (ct as CodeNamespace).Members;
  64.     }
  65.     else
  66.     {

  67.         elements = (ct as CodeClass).Members;
  68.     }

  69.     foreach (var ce in elements)
  70.     {
  71.         if ((ce is CodeNamespace) | (ce is CodeClass))
  72.         {
  73.             //寻找嵌套的类
  74.             SetAllClass(ce as CodeElement, str);
  75.         }
  76.     }
  77. }
  78. /// <summary>
  79. /// 为类添加工厂方法。
  80. /// 这个函数主要演示了:
  81. /// 1、如何用AddFunction方法为类添加方法。
  82. /// 2、CodeModel和文档操作之间的结合。
  83. /// code***.GetStartPoint/GetEndPoint可以获得TextPoint;
  84. /// TextPoint.CodeElement可获得对应的codeElement。
  85. /// 3、为不同的编程语言(VB.NET,c#)提供插件
  86. /// </summary>
  87. /// <param name="cc">要添加方法的类</param>
  88. /// <param name="fullname">基类的全称</param>
  89. private void AddFactoryMethod(CodeClass cc, string fullname)
  90. {

  91.     string str1 = "";
  92.     string str2 = "";

  93.     if (cc.Language == CodeModelLanguageConstants.vsCMLanguageCSharp)
  94.     {
  95.         str1 = string.Format("return new {0}();\n", cc.Name);
  96.         if (cc.FullName == fullname)
  97.             //基类
  98.             str2 = "public virtual";
  99.         else//子类
  100.             str2 = "public override";

  101.     }
  102.     else if (cc.Language == CodeModelLanguageConstants.vsCMLanguageVB)
  103.     {
  104.         str1 = string.Format("return new {0}()\n", cc.Name);
  105.         if (cc.FullName == fullname)
  106.             //基类
  107.             str2 = "Public Overridable";
  108.         else//子类
  109.             str2 = "Public Overrides";
  110.     }
  111.     //添加函数
  112.     CodeFunction cf = cc.AddFunction("GetInstance", vsCMFunction.vsCMFunctionFunction,
  113.         fullname, -1, vsCMAccess.vsCMAccessPublic, null);
  114.     //为函数添加文档注释
  115.     cf.DocComment = "<summary>\n工厂方法生成一个实例。\n</summary>";
  116.     EditPoint ep = cf.GetEndPoint(vsCMPart.vsCMPartBody).CreateEditPoint();
  117.     //添加函数体
  118.     ep.Insert(str1);
  119.     ep = cf.GetStartPoint(vsCMPart.vsCMPartHeader).CreateEditPoint();
  120.     ep.ReplaceText(6, str2, 0);
  121. }
  122. private DTE2 _applicationObject;
复制代码
完整代码可以在这里下载: 附件AddFactoryMethod.rar (下载  0 次)。

3、一些问题

3.1 CodeClass的属性DerivedTypes没有实现。否则寻找指定类的所有派生类将更为简化,程序更简单。

3.2 CodeClass.AddFunction方法,第2和第5个参数,使用示例代码以外的参数值,均会出现异常。Bug??所以示例最后被迫使用ReplaceText来完成加入虚函数的任务。希望有高手指点一下。

3.3 CodeModel中大部分集合,特别是有item属性但item非默认属性者,使用下标索引时,从1而非0开始。EnvDTE命名空间中其他集合大概也是。或者,推而广之,涉及到VSTA开发的(如VSTO),集合一般下标从1开始。毕竟这些领域还是VB为主。

3.4 和本文无关——后台编辑器在编辑代码的时候很不好用啊。尤其选择折叠代码的时候,居然不能正常显示。。。我浏览器的问题麽?

4、参考文献

1、《 Dynamic Objects, Factories, and Runtime Machines to Boost Performance》  by  Philip Liebscher
2、《 Creating a C# Class using the CodeModel Object》 MSDN
3、《 如何:使用 CodeModel 对象分析 Visual Basic 代码》 MSDN
4、关于Vs扩展的系列文章: Visual Studio 2008 可扩展性开发(九):总结篇
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值