如何强制MSTEST测试方法在运行前重置所有singletons/statics

本文关键字:singletons statics MSTEST 何强制 测试方法 运行 | 更新日期: 2023-09-27 17:48:53

我在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
    }

尽管在这个话题上出现了类似的问题,但它只是澄清了测试不是并行运行的。我意识到测试是串行运行的,但似乎没有办法明确地为每个方法强制使用一个新的AppDomain(或者类似于清除所有状态的方法)。

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

如何强制MSTEST测试方法在运行前重置所有singletons/statics

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

MSDN论坛上的这篇文章展示了如果您只有一些需要重置的静态信息,如何处理这种情况。它确实提到了一些选项(例如使用Reflection和PrivateType)。

我仍然欢迎任何进一步的想法,特别是如果我错过了MSTEST的一些明显的东西。

在测试中添加一个使用反射来删除singleton实例的助手(您也可以向singleton添加一个重置方法,但我会担心它的使用)。类似于:

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

在TestInitialize方法中调用此函数。[我知道这是"清理世界",但你只需要在每个单例的助手中写一次方法,它非常琐碎,并给你显式的控制]

我认为您正在寻找TestIntialize属性和TestCleanUp属性。这是一个MSDN博客,显示了执行顺序链接文本

我们的MSTest也出现了类似的问题。我们通过在需要它的特定测试的开始和结束时调用一个函数来处理它

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

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