我正在使用一些遗留代码,我必须在其中实现一个新的处理程序。不幸的是,在这个处理程序中,有一个对象被框架使用一些硬编码的属性文件初始化,如下所示:
// new code
public class NewHandler extends RootHandler {
Util util = Util.getInstance(); // !!!Problem: throws NPE in unit-testing env
// defined in the legacy framework, can't change its signature
@Override
public void doSth() {
}
}
// legacy code
public class Util {
static public synchronized Util getInstance() {
if (_instance == null) {
_instance = new Util();
}
return _instance;
}
private Util() {
MyObj obj = LegacyCode.load("myConfig.cfg"); // !!!Problem: throws NPE in unit-testing env
...
}
}
现在的问题是:
在 Util() 中,它总是从固定位置加载配置文件,因此当我运行单元测试时它总是抛出 NPE,因为单元测试类路径上没有“myConfig.cfg”。
一种解决方法是将业务逻辑提取到一个单独的方法中,传入 Util 对象,以便我可以在测试期间传入模拟对象:
// new code
public class NewHandler extends RootHandler {
Util util = null;
// defined in the legacy framework, can't change its signature
@Override
public void handle() {
Util util = util.getInstance();
doHandle(util);
}
public void doHandle(Util util) {
...
}
}
// legacy code
public class Util {
...
}
我的问题是:
有没有其他方法可以解决这个问题,而不添加 doHandle() 方法?我使用 Mockito 尝试了以下代码:
Util myUtil = mock(Util.class);
when(Util.getInstance()).thenReturn(myUtil);
但它不起作用,Util.getInstance() 没有被 myUtil 替换。
有什么想法吗?
更新:事情变得有点困惑,因为我发现 Util.getInstance() 在遗留代码库中随处被调用。因此,即使我的新代码中有一个模拟的 Util(正如下面 Stas 所描述的),当在其他地方调用 Util.getInstance() 时,事情仍然会中断。
显然,我无法向调用 Util.getInstance() 的遗留代码中的每个类添加新的构造函数。
这就是我问“是否可以在 Mockito 中添加诸如 when(Sth.getInstance()).thenReturn(myMock)”
的原因
如果是,则对 Util.getInstance() 的所有调用都可以被模拟,并且不会再引起问题。
嗯...有什么想法吗?
********************************************************************************
解决方案:我认为我的案例的解决方案是 PowerMock
最佳答案
你应该尝试 smt 就像
public class NewHandler extends RootHandler {
Util util;
//Pass instance
NewHandler(Util util) {this.util = util;}
}
您将能够在测试中使用它 new NewHandler(Util.getInstance())
或 new NewHandler(mock(Util.class))
。
PS 要了解更多信息,您可以阅读 Service locator和 DI
关于java - 这在 Mockito : when(SomeClass. getIntance()).thenReturn(myMock) 中可能吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5714387/