但是 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("验证此测试方法的正确性。");
}