[英]java: how to mock Calendar.getInstance()?
在我的代码中,我有这样的东西:
private void doSomething() {
Calendar today = Calendar.getInstance();
....
}
如何在我的 junit 测试中“模拟”它以返回特定日期?
你可以使用PowerMock和Mockito一起模拟它:
在课堂上:
@RunWith(PowerMockRunner.class)
@PrepareForTest({ClassThatCallsTheCalendar.class})
成功的关键是你必须将类放在PrepareForTest中使用Calendar而不是Calendar本身,因为它是一个系统类。 (在我发现这个之前,我个人不得不经常搜索)
然后是嘲弄本身:
mockStatic(Calendar.class);
when(Calendar.getInstance()).thenReturn(calendar);
据我所知,你有三个合理的选择:
以您在当天设置的任何方法/类中注入Calendar
实例。
private void method(final Calendar cal) { Date today = cal.getTime(); }
使用JodaTime而不是Calendar
。 这不是一个选择,而是一个建议的案例,因为JodaTime将使您的生活更轻松。 您仍需要将此时间注入方法。
DateTime dt = new DateTime();
Date jdkDate = dt.toDate();
在一些允许您获取时间的界面中包装Calendar
。 然后,您只需模拟该接口并使其返回一个常量Date
。
Date today = calendarInterfaceInstance.getCurrentDate()
不要嘲笑它 - 而是引入一个你可以模拟得到日期的方法。 像这样的东西:
interface Utility {
Date getDate();
}
Utilities implements Utility {
public Date getDate() {
return Calendar.getInstance().getTime();
}
}
然后你可以将它注入到你的类中,或者只使用带有一堆静态方法的helper类和一个load方法:
public class AppUtil {
private static Utility util = new Utilities();
public static void load(Utility newUtil) {
this.util = newUtil;
}
public static Date getDate() {
return util.getDate();
}
}
然后在您的应用程序代码:
private void doSomething() {
Date today = AppUtil.getDate();
....
}
然后,您可以在测试方法中加载模拟接口。
@Test
public void shouldDoSomethingUseful() {
Utility mockUtility = // .. create mock here
AppUtil.load(mockUtility);
// .. set up your expectations
// exercise the functionality
classUnderTest.doSomethingViaAPI();
// ... maybe assert something
}
使用Mockito和PowerMockito:
Calendar endOfMarch = Calendar.getInstance();
endOfMarch.set(2011, Calendar.MARCH, 27);
PowerMockito.mockStatic(Calendar.class);
Mockito.when(Calendar.getInstance()).thenReturn(endOfMarch);
有关完整代码,请参阅链接 。
编写一个叫做类DateHelper
与方法getCalendar
返回Calendar.getInstance()
重构您正在测试的类,使其具有DateHelper
类型的成员变量,以及注入该成员变量的构造函数。 使用该构造在您的测试,注入的模拟DateHelper
,其中getCalendar
已存根返回一些已知的日期。
你可以使用JMockit进行模拟。 在这里,您可以看到如何做到这一点: 模拟Java日历 - JMockit与Mockito 。
避免使用Calendar.getInstance()
并仅使用Mockito
方法返回您喜欢的内容。 例如:
@Test
fun italianLocale_returnsItalianFormatDate() {
val calendar: Calendar = Mockito.mock(Calendar::class.java)
Mockito.`when`(calendar.get(Calendar.DAY_OF_MONTH)).thenReturn(27)
Mockito.`when`(calendar.get(Calendar.YEAR)).thenReturn(2023)
Mockito.`when`(calendar.get(Calendar.MONTH)).thenReturn(1)
val formatted = calendar.toReadableDate()
assert(formatted == "27/01/2023")
}
在您的 gradle 文件中导入 Mockito:
testImplementation ("org.mockito.kotlin:mockito-kotlin:x.x.x")
或者(如果你使用的是 groovy)
testImplementation "org.mockito.kotlin:mockito-kotlin:x.x.x"
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.