我最近偶然发现了这个有趣的概念,它可以为我节省很多测试工作。我不明白的是如何在运行时注入提供程序?
场景很简单:我正在使用我选择的模拟框架在运行时构建一个模拟对象,但我不知道生成的类的名称,因为它是一个模拟(所以我不能提前配置它,我不想)。
有人在单元测试中成功地使用了这种技术吗?
谢谢你。
我最近偶然发现了这个有趣的概念,它可以为我节省很多测试工作。我不明白的是如何在运行时注入提供程序?
场景很简单:我正在使用我选择的模拟框架在运行时构建一个模拟对象,但我不知道生成的类的名称,因为它是一个模拟(所以我不能提前配置它,我不想)。
有人在单元测试中成功地使用了这种技术吗?
谢谢你。
由于使用了静态属性和服务定位器,这种模式对于单元测试非常不方便。为了能够运行验证使用此单例的代码的测试,您需要设置一个有效的服务定位器并使用您关心使用测试的单例(可能是模拟实例)对其进行配置。
即使是文章给出的示例也已经存在这些问题,因为“你喜欢单身人士吗?” 代码,很难测试:
if (DialogDisplayer.getDefault().yesOrNo(
"Do you like singletons?"
)) {
System.err.println("OK, thank you!");
} else {
System.err.println(
"Visit http://singletons.apidesign.org to"
+ " change your mind!"
);
}
更好的选择是使用构造函数注入来注入该单例(请原谅我的法语,但我不是母语 Java):
public class AskTheUserController
{
private DialogDisplayer dialogDisplayer;
private MessageDisplayer messageDisplayer;
public AskTheUserController(DialogDisplayer dialogDisplayer,
MessageDisplayer messageDisplayer)
{
this.dialogDisplayer = dialogDisplayer;
this.messageDisplayer = messageDisplayer;
}
public void AskTheUser()
{
if (this.dialogDisplayer.yesOrNo(
"Do you like singletons?"
)) {
this.messageDisplayer.display("OK, thank you!");
} else {
this.messageDisplayer.display(
"Visit http://singletons.apidesign.org to"
+ " change your mind!"
);
}
}
}
该代码中还有另一个“隐藏”依赖项:System.err.println
. 它使用MessageDisplayer
接口进行了抽象。这段代码有几个明显的优点:
您的测试可能如下所示:
@Test
public void AskTheUser_WhenUserSaysYes_WeThankHim()
{
// Arrange
bool answer = true;
MockMessageDisplayer message = new MockMessageDisplayer();
MockDialogDisplayer dialog = new MockDialogDisplayer(answer);
AskTheUserController controller =
new AskTheUserController(dialog, message);
// Act
controller.AskTheUser();
// Assert
Assert.AreEqual("OK, thank you!", message.displayedMessage);
}
@Test
public void AskTheUser_WhenUserSaysNo_WeLetHimChangeHisMind()
{
// Arrange
bool answer = true;
MockMessageDisplayer message = new MockMessageDisplayer();
MockDialogDisplayer dialog = new MockDialogDisplayer(answer);
AskTheUserController controller =
new AskTheUserController(dialog, message);
// Act
controller.AskTheUser();
// Assert
Assert.IsTrue(
message.displayedMessage.contains("change your mind"));
}
当您使用文章中所示的“可注入单例”模式时,您的测试代码将永远不会像上面的代码那样显示。
单例没有错,单例在任何软件中都是有用且必要的概念。问题是您不应该使用静态字段和方法来实现它们。
我使用 Guice 来注入我的单例,并且很长一段时间以来我都不必在我的代码库和测试中使用静态。
以下是一些您可能会觉得有用的链接,它们解释了如何使用 Guice 实现可测试的单例: