如何在运行之前强制MSTEST TestMethod重置所有单例/静态?

时间:2008-10-10 21:45:52

标签: c# unit-testing visual-studio-2008 mstest appdomain

我在Visual Studio 2008中使用MSTEST。如何在某个测试类中使用每个单元测试方法,就好像它是第一个运行的测试,以便在运行每个测试之前重置所有全局状态?我不想使用TestInitialize,ClassInitialize,AssemblyInitialize等明确清理世界。例如:

[TestClass]
public class MyClassTests
{
    [TestMethod]
    public void Test1()
    {
       // The "Instance" property creates a new instance of "SomeSingleton"
       // if it hasn't been created before.
       var i1 = SomeSingleton.Instance;
       ...
    }
    [TestMethod]
    public void Test2()
    {
       // When I select "Test1" and "Test2" to run, I'd like Test2
       // to have a new AppDomain feel so that the static variable inside
       // of "SomeSingleton" is reset (it was previously set in Test1) on
       // the call to ".Instance"
       var i2 = SomeSingleton.Instance;
       // some code
    }

虽然此主题出现了similar question,但它只是澄清了测试并不是并行运行的。我意识到测试是连续运行的,但似乎没有办法为每个方法显式强制使用新的AppDomain(或等同于清除所有状态的东西)。

理想情况下,我只想为单元测试的一小部分指定此行为,这样我就不必为不关心全局状态的测试支付新AppDomain创建的代价(绝大多数我的测试。)

4 个答案:

答案 0 :(得分:7)

最后,我编写了一个使用AppDomain.CreateDomain的帮助器,然后使用反射在不同的AppDomain下调用单元测试。它提供了我需要的隔离。

MSDN论坛上的

This post显示了如果只有一些需要重置的静态,如何处理这种情况。 It确实提到了一些选项(例如使用反射和PrivateType)。

我继续欢迎任何进一步的想法,特别是如果我遗漏了关于MSTEST的明显事实。

答案 1 :(得分:6)

在你的测试中添加一个帮助器,它使用反射来删除单例实例(你也可以为单例添加一个重置方法,但我会担心它的使用)。类似的东西:

public static class SingletonHelper {
            public static void CleanDALFactory() 
            {
                    typeof(DalFactory)
                        .GetField("_instance",BindingFlags.Static | BindingFlags.NonPublic)
                        .SetValue(null, null);
            }
}

在TestInitialize方法中调用它。 [我知道这是“清理世界”,但你只需要在每个单独的帮助器中编写一次方法,这非常简单并且可以让你明确控制]

答案 2 :(得分:2)

我认为您正在寻找TestIntialize属性和TestCleanUp属性。这是一个显示执行顺序link text

的MSDN博客

答案 3 :(得分:1)

我们的MSTests出现了类似的问题。我们通过在需要它的特定测试的开头和结尾调用函数来处理它。

我们正在应用配置中存储测试过期日期。需要在此日期进行三次测试,以确定适当的值。我们的应用程序设置方式,只有在会话中没有分配值时才会重置配置值。因此,我们创建了两个新的私有静态函数 - 一个用于将配置值显式设置为指定日期,另一个用于在测试运行后从会话中清除该日期。在我们的三个测试中,我们称这两个函数。当下一个测试运行时,应用程序会看到日期的空值,并从配置文件中重新获取它。

我不确定这是否有用,但这就是我们解决类似问题的方法。