网上找的大多不好用。甚至有的答案说不能。
本人不服,试了一个晚上终于调试成功了。
@RunWith(PowerMockRunner.class)
@PrepareForTest({EventResolverImpl.class})
public class EventTypeResolverTest {
static {
PowerMockito.mockStatic(System.class);
PowerMockito.when(System.getenv("CF_ORG")).thenReturn("dm-canary");
PowerMockito.when(System.getenv("CF_SPACE")).thenReturn("dev");
PowerMockito.when(System.getenv("CF_DOMAIN")).thenReturn("cfapps.sap.hana.ondemand.com");
}
说明一点相关原理。
这个 PrepareForTest 注解可以在跑TEST CASE前预处理被测试的类,通常这个类可能是个IMPL,你可能测试的是controller中的方法,最终会调用到这个类中的业务逻辑。我们要做的是跑CASE之前把impl里面通常是final String的变量prepare出来。比如这种:
private static final String CF_DOMAIN = System.getenv("CF_DOMAIN");
经过我打断点进行尝试,我发现被测试类的final变量初始化代码,优先于 测试类中@Before注释的初始化方法。所以我们的mock规则写在@Before中是没用的。
看到这儿我就感觉这属于编译优先级的知识了。问题变成都是静态代码,如何调整优先级。所以想到测试类中使用static代码块保证mock规则优于 PrePare执行。
结论:测试类置顶static代码块优先级>被测试类注入的impl类的final变量初始化优先级> 测试类中@Before注解标注的初始化代码优先级>被测试类的测试逻辑优先级
测试方法中直接跑就行,由于我们在static代码中提前mock好了静态数据。当case运行时就不会出现环境变量拿到NULL从而导致的空指针问题。