首次尝试任何真实的单元测试。我有一个WPF客户端应用程序,它从WCF服务中的方法接收数据。这些方法调用直接来自客户端应用程序中的视图模型:
public string LoadArticle()
{
MyServiceClient msc = new MyServiceClient();
return Article = msc.GetArticle(_UserName);
}
然后我有一个测试项目,我在其中创建一个viewmodel然后调用我的方法:
[TestMethod]
public void LoadArticleTestValid()
{
var articleViewModel = new ArticleViewModel("Thomas");
string userName = "Thomas";
string expected = "Article 1";
var actual = articleViewModel.LoadArticle(userName);
etc.
}
显然,此测试将失败,因为客户端应用程序无法访问服务以调用LoadArticle
。这种情况是如何解决的?我是否使用依赖注入并将某种类型的IMyService接口传递给构造函数,而不是在ViewModel中创建MyServiceClient,还是以某种方式模拟服务?
答案 0 :(得分:3)
这是问题所在:
MyServiceClient msc = new MyServiceClient();
这会在ArticleViewModel
和MyServiceClient
之间产生紧密耦合。为了单元测试只是 ArticleViewModel
,需要模拟这种依赖。如果有一个IMyServiceClient
,那么你可以将它提供给班级:
public ArticleViewModel
{
private IMyServiceClient ServiceClient { get; set; }
public ArticleViewModel(IMyServiceClient serviceClient)
{
this.ServiceClient = serviceClient;
}
// etc.
}
然后该类中的代码不会创建新的服务客户端,它只会使用该属性中的代码。
然后在单元测试中,您将创建一个模拟IMyServiceClient
,在该模拟上定义预期行为,并将其提供给正在测试的对象。你如何做到这一点取决于模拟库。 Rhino Mocks中的简单示例可能如下所示:
// arrange
var serviceClient = MockRepository.GenerateMock<IMyServiceClient>();
serviceClient.Stub(c => c.GetArticle(Arg<string>.Is.Anything)).Return("Article 1");
var model = new ArticleViewModel(serviceClient);
// act
var result = model.LoadArticle("Thomas");
// assert
Assert.AreEqual("Article 1", result);
这里的想法是单元测试仅测试LoadArticle
方法,而不是该方法调用的依赖项。这些依赖项提供了预期的行为,并根据这些预期的行为检查结果。
注意:没有什么可以阻止单元测试提供MyServiceClient
的实际实现,而不是模拟。单元测试项目只需要该服务的配置即可运行。 (单元测试项目是应用程序主机,它们可以有App.config
个文件。)这可能是集成测试而不是单元测试,但可以对结果进行相同的断言。
答案 1 :(得分:1)
是的,我认为你是对的我会建议一个IMyService的构造函数参数,你可以用它来将一个模拟注入到对象中进行测试。
此外!我建议不要使用自动生成的服务客户端。如果您将代码复制并粘贴到您自己的类中,您可以使其实现IMyService并有效地隐藏它使用WCF的事实,直接转到数据库或者是真实代码中的模拟对象。
这将允许您将Mock注入WPF以进行UI测试