testable-mock/docs/zh-cn/doc/use-mock.md
2020-11-22 22:12:51 +08:00

5.0 KiB
Raw Blame History

快速Mock被测类的任意方法调用

1. 覆写任意类的方法调用

在测试类里定义一个有@TestableMock注解的普通方法,使它与需覆写的方法名称、参数、返回值类型完全一致,然后在其参数列表首位再增加一个类型为该方法原本所属对象类型的参数。

此时被测类中所有对该需覆写方法的调用将在单元测试运行时将自动被替换为对上述自定义Mock方法的调用。

注意:也可以将需覆写的方法名写到@TestableMock注解的targetMethod参数里这样Mock方法自身就可以随意命名了当遇到重名的待覆写方法时特别有用

例如,被测类中有一处"anything".substring(1, 2)调用,我们希望在运行测试的时候将它换成一个固定字符串,则只需在测试类定义如下方法:

// 原方法签名为`String substring(int, int)`
// 调用此方法的对象`"anything"`类型为`String`
// 则Mock方法签名在其参数列表首位增加一个类型为`String`的参数(名字随意)
// 此参数可用于获得当时的实际调用者的值和上下文
@TestableMock
private String substring(String self, int i, int j) {
    return "sub_string";
}

完整代码示例见java-demokotlin-demo示例项目中的should_able_to_mock_common_method()测试用例。(由于Kotlin对String类型进行了魔改故Kotlin示例中将被测方法在BlackBox类里加了一层封装)

2. 覆写被测类自身的成员方法

有时候在对某些方法进行测试时希望将被测类自身的另外一些成员方法Mock掉。

操作方法与前一种情况相同Mock方法的第一个参数类型需与被测类相同即可实现对被测类自身不论是公有或私有成员方法的覆写。

例如,被测类中有一个签名为String innerFunc(String)的私有方法,我们希望在测试的时候将它替换掉,则只需在测试类定义如下方法:

// 被测类型是`DemoMock`
// 因此在定义Mock方法时在目标方法参数首位加一个类型为`DemoMock`的参数(名字随意)
@TestableMock
private String innerFunc(DemoMock self, String text) {
    return "mock_" + text;
}

完整代码示例见java-demokotlin-demo示例项目中的should_able_to_mock_member_method()测试用例。

3. 覆写任意类的静态方法

对于静态方法的Mock与普通方法相同。但需要注意的是对于静态方法传入Mock方法的第一个参数实际值始终是null

例如,在被测类中调用了BlackBox类型中的静态方法secretBox(),改方法签名为BlackBox secretBox()则Mock方法如下

// 目标静态方法定义在`BlackBox`类型中
// 在定义Mock方法时在目标方法参数首位加一个类型为`BlackBox`的参数(名字随意)
// 此参数仅用于标识目标类型,实际传入值将始终为`null`
@TestableMock
private BlackBox secretBox(BlackBox ignore) {
    return new BlackBox("not_secret_box");
}

完整代码示例见java-demokotlin-demo示例项目中的should_able_to_mock_static_method()测试用例。

4. 覆写任意类的new操作

在测试类里定义一个有@TestableMock注解的普通方法,将注解的targetMethod参数写为"",然后使该方法与要被创建类型的构造函数参数、返回值类型完全一致,方法名称随意。

此时被测类中所有用new创建指定类的操作并使用了与Mock方法参数一致的构造函数将被替换为对该自定义方法的调用。

例如,在被测类中有一处new BlackBox("something")调用希望在测试时将它换掉通常是换成Mock对象或换成使用测试参数创建的临时对象则只需定义如下Mock方法

// 要覆写的构造函数签名为`BlackBox(String)`
// 无需在Mock方法参数列表增加额外参数由于使用了`targetMethod`参数Mock方法的名称随意起
// 此处的`CONSTRUCTOR`为`TestableTool`辅助类提供的常量,值为"<init>"
@TestableMock(targetMethod = CONSTRUCTOR)
private BlackBox createBlackBox(String text) {
    return new BlackBox("mock_" + text);
}

完整代码示例见java-demokotlin-demo示例项目中的should_able_to_mock_new_object()测试用例。

5. 识别当前测试用例和调用来源

在Mock方法中可以通过TestableTool.TEST_CASETestableTool.SOURCE_METHOD来识别当前运行的测试用例名称进入该Mock方法前的被测类方法名称,从而区分处理不同的调用场景。

完整代码示例见java-demokotlin-demo示例项目中的should_able_to_get_source_method_name()should_able_to_get_test_case_name()测试用例。

6. 验证Mock方法被调用的顺序和参数

在测试用例中可用通过TestableTool.verify()方法,配合with()withInOrder()without()withTimes()等方法实现对Mock调用情况的验证。

详见校验Mock调用文档。