15

我有这个类我想单元测试:

public class SomeClass
{
    public void Foo()
    {
        Bar();
    }

    private void Bar()
    {
        Task.Factory.StartNew(() =>
        {
            // Do something that takes some time (e.g. an HTTP request)
        });
    }
}

这就是我的单元测试的样子:

[TestMethod]
public void TestFoo()
{
    // Arrange
    var obj = new SomeClass();

    // Act
    obj.Foo();
    obj.Foo();
    obj.Foo();

    // Assert
    /* I need something to wait on all tasks to finish */
    Assert.IsTrue(...);
}

所以,我需要让单元测试线程等到Bar方法中启动的所有任务都完成它们的工作,然后再开始我的断言。

重要提示:我无法改变SomeClass

我怎样才能做到这一点?

4

3 回答 3

27

解决此问题的一种方法是定义您自己的任务调度程序,使您可以跟踪嵌套任务的完成情况。例如,您可以定义一个同步执行任务的调度程序,如下所示:

class SynchronousTaskScheduler : TaskScheduler
{
    protected override void QueueTask(Task task)
    {
        this.TryExecuteTask(task);
    }

    protected override bool TryExecuteTaskInline(Task task, bool wasPreviouslyQueued)
    {
        return this.TryExecuteTask(task);
    }

    protected override IEnumerable<Task> GetScheduledTasks()
    {
        yield break;
    }
}

随后,创建此同步任务调度程序的一个实例,并使用它来执行一个根任务,该任务反过来会产生所有“隐藏”任务。由于嵌套任务从其父任务继承当前任务调度程序,所有内部任务也将在我们的同步调度程序上运行,这意味着我们最外层的StartNew调用只会在所有任务完成后返回。

TaskScheduler scheduler = new SynchronousTaskScheduler();

Task.Factory.StartNew(() =>
{
    // Arrange
    var obj = new SomeClass();

    // Act
    obj.Foo();
    obj.Foo();
    obj.Foo();
}, 
    CancellationToken.None,
    TaskCreationOptions.None,
    scheduler);

// Assert
/* I need something to wait on all tasks to finish */
Assert.IsTrue(...);

这种方法的一个缺点是您将失去任务的所有并发性;但是,您可以通过将自定义调度程序增强为一个并发但仍允许您跟踪正在执行的任务来解决此问题。

于 2013-07-24T21:10:32.030 回答
6
Task.WaitAll(the, list, of, task, objects, you, need, to, wait, on);

如果这是一种void async方法,那么您将无法做到。设计坏了。他们只是为了火而忘记。

于 2013-07-24T20:28:31.390 回答
3

不确定您是否可以进行此更改,但我可以这样做:

namespace ParallelProgramming.Playground
{
    public class SomeClass
    {
        public Task Foo()
        {
            return Bar();
        }

        private static Task Bar()
        {
            return Task.Factory.StartNew(() =>
                {
                    Console.WriteLine("I fired off. Thread ID: {0}", Thread.CurrentThread.ManagedThreadId);
                    Thread.Sleep(5000);
                    return true; //or whatever else you want.
                });
        }
    }

    [TestClass]
    public class StackOverflow
    {
        [TestMethod]
        public void TestFoo()
        {
            // Arrange
            var obj = new SomeClass();

            var results = new ConcurrentBag<Task>(); 
            var waitForMe = Task.Factory.StartNew(() =>
                {
                    // Act
                    results.Add(obj.Foo());
                    results.Add(obj.Foo());
                    results.Add(obj.Foo());

                    return true;
                });


            Task.WaitAll(waitForMe);

            // Assert
            /* I need something to wait on all tasks to finish */
            Assert.IsTrue(waitForMe.Result);
            Assert.AreEqual(3, results.Count);
        }
    }
}
于 2013-07-24T20:47:36.867 回答