[英]Async CTP, unit testing a ViewModel's async Method
我有一个单元测试(使用MSTest),如下所示:
[TestMethod]
public void MyTest()
{
var viewModel = new MyViewModel();
viewModel.Run();
//Assert something here
}
Run是一个返回void的异步方法。
让我们说Run
就像这样实现:
public async void Run()
{
//Show a busy indicator here
try
{
var result = await myAsyncModelClass.LongRunningOperation();
//Use the results here
}
finally
{
//Hide the busy indicator here
}
}
myAsyncModelClass.LongRunningOperation()
,本身就是一个异步方法,它返回一些Task<T>
,其中T是我的ViewModel感兴趣的结果。
我的问题是,我的测试是异步运行Run
方法,因此在Run
方法完成之前调用我的断言。 奇怪的是,当我放置一个断点时,b / c永远不会达到finally块,因为断言失败了。 如何保持Run
方法同步以便能够对其进行单元测试?
我也有myAsyncModelClass.LongRunningOperation()
的单元测试,但我只是调用Task<T>.Wait()
因为它返回一个任务。 这使得它在单元测试时同步。
另外,我想提一下, Run()
由一个MVVM框架神奇地调用ICommand。 void
可能是也可能不是require返回类型,我将不得不尝试一下。
异步方法需要一个“返回”的上下文。 由于MSTests在线程池上运行,因此默认情况下,异步方法也在线程池线程上继续(并且不阻止MSTest方法)。
在(C# Testing) Unit Testing
示例(在Async CTP安装目录中)下,有一个名为GeneralThreadAffineContext
的类型,可以这样使用:
[TestMethod]
public void MyTest()
{
MyViewModel viewModel = null;
GeneralThreadAffineContext.Run(() =>
{
viewModel = new MyViewModel();
viewModel.Run();
});
//Assert something here
}
还有特定的WPF和WinForms上下文,但线程仿射上下文应该适用于一般的ViewModel(不明确使用Dispatcher
)。
更新2012-02-05:如果您可以更改ViewModel方法以返回Task
,那么您还有另一个选项:新的AsyncUnitTests库 。 安装NuGet包,将TestClass
更改为AsyncTestClass
,并且可以更自然地编写异步单元测试:
[TestMethod]
public async void MyTest()
{
MyViewModel viewModel = new MyViewModel();
await viewModel.Run();
//Assert something here
}
更新2012-09-04: Visual Studio 2012包含async
单元测试,因此您不再需要AsyncUnitTests
库:
[TestMethod]
public async Task MyTest()
{
MyViewModel viewModel = new MyViewModel();
await viewModel.Run();
//Assert something here
}
由于Visual Studio 2012 MSTest支持异步测试方法。 记住他们应该返回Task而不是void:
[TestMethod]
public async Task MyTest()
{
MyViewModel viewModel = new MyViewModel();
await viewModel.Run();
//Assert something here
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.