mirror of
https://github.com/alibaba/testable-mock.git
synced 2025-01-10 20:30:11 +08:00
90 lines
3.1 KiB
Markdown
90 lines
3.1 KiB
Markdown
测试无返回值的方法
|
||
---
|
||
|
||
如何对void类型的方法进行测试一直是许多单元测试框架在悄悄回避的话题,由于以往的单元测试手段主要是对被测单元的返回结果进行校验,当遇到方法没有返回值时就会变得无从下手。
|
||
|
||
从功能的角度来说,虽然void方法不返回任何值,但它的执行一定会对外界产生某些潜在影响,我们将其称为方法的"副作用",比如:
|
||
|
||
1. 初始化某些外部变量(私有成员变量或者全局静态变量)
|
||
2. 在方法体内对外部对象实例进行赋值
|
||
3. 输出了日志
|
||
4. 调用了其他外部方法
|
||
5. ... ...
|
||
|
||
|
||
> 不返回任何值也不产生任何"副作用"的方法没有存在意义。
|
||
|
||
这些"副作用"归纳来说可分为两类:**修改外部变量**和**调用外部方法**。
|
||
|
||
通过TestableMock的私有字段访问和Mock校验器可以很方便的实现对"副作用"的结果检查。
|
||
|
||
#### 修改外部变量的void方法
|
||
|
||
例如,下面这个方法会根据输入修改私有成员变量`hashCache`:
|
||
|
||
```java
|
||
class Demo {
|
||
private Map<String, Integer> hashCache = mapOf();
|
||
|
||
public void updateCache(String domain, String key) {
|
||
String cacheKey = domain + "::" + key;
|
||
Integer num = hashCache.get(cacheKey);
|
||
hashCache.put(cacheKey, count == null ? initHash(key) : nextHash(num, key));
|
||
}
|
||
|
||
... // 其他方法省略
|
||
}
|
||
```
|
||
|
||
若要测试此方法,可以利用TestableMock直接读取私有成员变量的值,对结果进行校验:
|
||
|
||
```java
|
||
class DemoTest {
|
||
private Demo demo = new Demo();
|
||
|
||
@Test
|
||
public void testSaveToCache() {
|
||
Integer firstVal = demo.initHash("hello"); // 访问私有方法
|
||
Integer nextVal = demo.nextHash(firstVal, "hello"); // 访问私有方法
|
||
demo.saveToCache("demo", "hello");
|
||
assertEquals(firstVal, demo.hashCache.get("demo::hello")); // 读取私有变量
|
||
demo.saveToCache("demo", "hello");
|
||
assertEquals(nextVal, demo.hashCache.get("demo::hello")); // 读取私有变量
|
||
}
|
||
}
|
||
```
|
||
|
||
#### 调用外部方法的void方法
|
||
|
||
例如,下面这个方法会根据输入打印信息到控制台:
|
||
|
||
```java
|
||
class Demo {
|
||
public void recordAction(Action action) {
|
||
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss ");
|
||
String timeStamp = df.format(new Date());
|
||
System.out.println(timeStamp + "[" + action.getType() + "] " + action.getTarget());
|
||
}
|
||
}
|
||
```
|
||
|
||
若要测试此方法,可以利用TestableMock快速Mock掉`System.out.println`调用,然后用`InvokeVerifier.verify()`方法校验传入的打印内容:
|
||
|
||
```java
|
||
class DemoTest {
|
||
private Demo demo = new Demo();
|
||
|
||
// 拦截`System.out.println`调用
|
||
@TestableMock
|
||
public void println(PrintStream ps, String msg) {}
|
||
|
||
@Test
|
||
public void testRecordAction() {
|
||
Action action = new Action("click", ":download");
|
||
demo.recordAction();
|
||
// 验证Mock方法`println`被调用,且传入参数符合预期
|
||
verify("println").matches("\\d{4}-\\d{2}-\\d{2} \\d{2}:\\d{2}:\\d{2} \\[click\\] :download");
|
||
}
|
||
}
|
||
```
|