VS2005对私有方法的单元进行测试

2010-08-28 10:48:27来源:西部e网作者:

  使用 Visual Studio 对公共方法进行单元测试,跟以前 NUnit 一样,都是引用需要的类,然后对公共方法进行测试。

  但是 Visual Studio 还可以对类的私有方法进行测试。而Visual Studio 不允许测试代码跟实际代码放在一个项目中,我们来看看是Visual Studio UnitTest如何做的。

  比如我们有这样一个私有方法

namespace ClassLibrary1{
 public class DivisionClass{
  private int Divide_private(int numerator, int denominator){
   return numerator / denominator;
  }
 }
}

  我们只要在这个私有方法的右键菜单中选择 创建单元测试,系统就自动产生了这个私有方法的单元测试代码。

  下面我们来分析产生的单元测试代码,看Visual Studio UnitTest 是如何对私有方法进行单元测试的

  简单来说,Visual Studio UnitTest 生成私有方法的单元测试时,将自动创建一个私有访问器。私有访问器是测试方法用于访问私有代码的方法。单元测试生成对私有访问器的调用,然后通过私有访问器来调用私有方法。私有访问器驻留在测试项目中的文件中;因此将被编译为测试项目程序集。

  具体来看测试项目:

  首先我们可以看到一个名为 VSCodeGenAccessors.cs 的新文件被创建,

  这个文件包含两个类:

  internal 类型的 BaseAccessor 类 和 派生自它的 ClassLibrary1_DivisionClassAccessor 类

  BaseAccessor 类 是通用的访问器基类。

  ClassLibrary1_DivisionClassAccessor 类 则是对你要访问类的私有方法进行了反射封装,这样你就可以通过操作这个类来操作该私有方法了。如下面代码:

using Microsoft.VisualStudio.TestTools.UnitTesting;

namespace TestProject1{
 [System.Diagnostics.DebuggerStepThrough()]
 [System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.TestTools.UnitTestGeneration", "1.0.0.0")]
 internal class BaseAccessor {

  protected Microsoft.VisualStudio.TestTools.UnitTesting.PrivateObject m_privateObject;

  protected BaseAccessor(object target, Microsoft.VisualStudio.TestTools.UnitTesting.PrivateType type) {
   m_privateObject = new Microsoft.VisualStudio.TestTools.UnitTesting.PrivateObject(target, type);
  }

  protected BaseAccessor(Microsoft.VisualStudio.TestTools.UnitTesting.PrivateType type) :
  this(null, type) {}

  internal virtual object Target {
   get {
    return m_privateObject.Target;
   }
  }

  public override string ToString() {
   return this.Target.ToString();
  }

  public override bool Equals(object obj) {
   if (typeof(BaseAccessor).IsInstanceOfType(obj)) {
    obj = ((BaseAccessor)(obj)).Target;
   }
   return this.Target.Equals(obj);
  }

  public override int GetHashCode() {
   return this.Target.GetHashCode();
  }
 }

 [System.Diagnostics.DebuggerStepThrough()]
 [System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.TestTools.UnitTestGeneration", "1.0.0.0")]
 internal class ClassLibrary1_DivisionClassAccessor : BaseAccessor {

 protected static Microsoft.VisualStudio.TestTools.UnitTesting.PrivateType m_privateType = new Microsoft.VisualStudio.TestTools.UnitTesting.PrivateType(typeof(global::ClassLibrary1.DivisionClass));

 internal ClassLibrary1_DivisionClassAccessor(global::ClassLibrary1.DivisionClass target) :
 base(target, m_privateType) {}

 internal int Divide_private(int numerator, int denominator) {
  object[] args = new object[] {
   numerator,
   denominator};
  int ret = ((int)(m_privateObject.Invoke("Divide_private", new System.Type[] {
   typeof(int),
   typeof(int)}, args)));
   return ret;
  } 
 }
}

  注意:当您更改正在测试的代码文件中的私有方法时,这个访问器可能无法正常工作,需要重新生成专用访问器(ClassLibrary1_DivisionClassAccessor 类)。

  整理一下就是:

  这个访问器,是通过反射的方式实现的。

  VSUT利用自动代码生成技术,在单元测试项目中先来给你要测试的类生成一个名字叫XXXAccessor的访问器。

  这个访问器会把需要测试类的需要测试的私有方法暴露出来,这种方式,无论是私有还是公共的属性和方法。这种方式可以很方便的给测试方法搭建测试环境,MOCK对象的注入也容易了。在测试调用的时候,就简单的只有下面的代码了:

[DeploymentItem("ClassLibrary1.dll")]
[TestMethod()]
public void Divide_privateTest()
{
 DivisionClass target = new DivisionClass();
 TestProject1.ClassLibrary1_DivisionClassAccessor accessor = new  TestProject1.ClassLibrary1_DivisionClassAccessor(target);
 int numerator = 4;
 int denominator = 0;
 int expected = 0;
 int actual;
 actual = accessor.Divide_private(numerator, denominator);
 Assert.AreEqual(expected, actual, "ClassLibrary1.DivisionClass.Divide_private 未返回所需的值。");
 Assert.Inconclusive("验证此测试方法的正确性。");
}
关键词:VS.net

赞助商链接: