重置 JUnit 测试的静态字段

2022-09-04 01:45:20

我有一组JUnit测试,它们调用Java程序上的main方法,传入args并检查输出。没关系。

但是,如果我正在测试的程序具有更改的静态值,则它们在测试之间将保持不变。这会导致问题。我无法控制正在测试的程序到底是什么,或者用于静态字段的名称。

我如何确保我的单元测试干净利落地运行,就好像它从头开始启动程序一样,而不保留这些静态字段。有没有办法以某种方式重置它们?

如果没有,我将不得不启动一个运行程序的新进程,然后检查输出等,但这似乎有点过分了。

编辑 - 请注意,我无法控制单元测试正在测试的代码 - 我无法更改它们的字段名称,不幸的是,我也不知道它们的字段名称。我在想,如果不开始新的过程,这是不可能的吗?


答案 1

您应该显式初始化测试类中的任何静态状态,通常在带注释或@Before@BeforeClass

这就是为什么在应用程序中具有大量静态依赖项对于测试来说是一个坏主意的原因。这就是为什么许多人鼓励无状态编程。


答案 2

一般来说,如果你发现你的代码是不可测试的,就像这里的问题是,这是代码气味的标志,你应该认真考虑重构你的代码,不要使用这些静态字段。

话虽如此,您可能会发现BeanInject库很有帮助。您可以将带注释的方法放入测试类中,并让它使用注入重置静态字段:@After

Inject.field("thatStaticField").of(thatObjectWithStaticFields).with("default value");

这样,您只需要知道字段名称,而不必实际修改具有字段的类。库使用反射来执行此操作。

此外,我想到,如果你正在测试的东西包含你无法控制的部分,你为什么不尝试用Mockito来嘲笑这些部分呢?

编辑/添加:好吧,所以你的问题是你甚至不知道类可能具有或可能没有的可能静态变量的初始值。我看到两种可能的方法:1)您必须在首次加载类时保存它们的值并在每次测试之间重置值,或者2)您必须从类装入器中获取类的全新实例。

在第 1 点上),您需要使用反射来遍历 @BeforeClass 方法中的所有字段,将其初始值保存到某个结构中,然后重置@Before或@After方法中的值。下面是一些有关使用反射遍历类的字段的主题:循环访问 Java 类中的所有字段Map<String,Object>

关于第2点),您有这方面的说明(涉及类装入器):Java:如何“重新启动”静态类?

你可以用反射和那些东西做什么,这很酷。:)


推荐