如何对从任务中派生的方法进行单元测试.ContinueWith



考虑以下代码:

public interface IBar
{
    Task<IEnumerable<string>> GetStringsAsync();
}
public class Foo
{
    public Foo(IBar bar, IList<string> initial)
    {
        MyCollection = new ObservableCollection<string>();
        if (initial == null || !initial.Any())
            AddContent(bar);
        else
            MyCollection.AddRange(initial);
    }
    public ObservableCollection<string> MyCollection { get; private set; }
    public void AddContent(IBar bar)
    {
        var cancel = new CancellationTokenSource();
        bar.GetStringsAsync().ContinueWith(
            task => MyCollection.AddRange(task.Result),
            cancel,
            TaskContinuationOptions.NotOnCancel,
            TaskScheduler.FromCurrentSynchronizationContext());
    }
}

如何对Foo.AddContent方法进行单元测试?我想测试一下,我的mock IBar提供的字符串实际上确实被添加到了集合中,但断言总是在任务更新集合之前被调用。

我使用的是.NET 4.5.2。我的第一个选择是在AddContent中使用asyncawait,但由于该方法是在构造函数中使用的,我认为最好避免这种情况。我需要一些可以启动异步数据加载的东西,但不会等待它完成。

关于如何重写AddContent的建议是受欢迎的,但我已经尝试了很多东西,这是唯一一个运行良好的,所以我真正想要的是一种测试它的方法。

更新2

使用Stephen Cleary的异步初始化模式。

http://blog.stephencleary.com/2013/01/async-oop-2-constructors.html

更新

由于问题已经改变,现在的要求是让构造函数接受变量IBar。根据传递给构造函数的IBar变量的硬性要求,我建议如下:

public class Foo
{
    public Foo(IBar bar)
    {
        MyCollection = new ObservableCollection<string>();
        MyCollection.AddRange(bar.GetStringsAsync().Result));
    }
    public ObservableCollection<string> MyCollection { get; private set; }
    public async Task AddContent(IBar bar)
    {
        MyCollection.AddRange(await bar.GetStringsAsync());
    }
}

注意:public方法仍然使用首选机制async / await,但构造函数只是调用.Result。这是构造函数中的阻塞调用,是一种非常糟糕的做法。这很容易被认为是你永远不应该做的事情…

我强烈建议您的构造函数将初始字符串改为这样(特别是考虑到它只用于返回的字符串!):

public class Foo
{
    public Foo(IEnumerable<string> strings)
    {
        MyCollection = new ObservableCollection<string>();
        MyCollection.AddRange(strings);
    }
    public ObservableCollection<string> MyCollection { get; private set; }
    public async Task AddContent(IBar bar)
    {
        MyCollection.AddRange(await bar.GetStringsAsync());
    }
}

用法

[TestMethod]
public async Task Test()
{
    IBar bar = GetMockedBarImpl();
    var sut = new Foo(await bar.GetStringsAsync());        
    Assert.IsTrue(sut.MyCollection.Any());
    // TODO: Add asserts for known strings in collection...
}

看起来有问题代码存在继承竞争条件。我会等待GetStringAsync完成并分配给MyCollection

public void AddContent(IBar bar)
{
    var cancel = new CancellationTokenSource();
    var result = bar.GetStringsAsync().ContinueWith(
        task => task.Result,
        cancel,
        TaskContinuationOptions.NotOnCancel,
        TaskScheduler.FromCurrentSynchronizationContext());
   MyCollection.AddRange(result.Result);
}

或简称

public void AddContent(IBar bar)
{
    var result = bar.GetStringsAsync().Result;
    MyCollection.AddRange(result);
}

最新更新