2020-10-23 23:22:53 +08:00
|
|
|
|
使用说明
|
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
## 引入Testable
|
|
|
|
|
|
2020-10-27 06:54:36 +08:00
|
|
|
|
首先在项目`pom.xml`文件中添加`testable-processor`依赖:
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
|
|
|
|
```xml
|
|
|
|
|
<dependency>
|
|
|
|
|
<groupId>com.alibaba.testable</groupId>
|
2020-10-27 06:54:36 +08:00
|
|
|
|
<artifactId>testable-processor</artifactId>
|
2020-10-23 23:22:53 +08:00
|
|
|
|
<version>${testable.version}</version>
|
|
|
|
|
<scope>provided</scope>
|
|
|
|
|
</dependency>
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
此时项目就获得了在单元测试中随意访问被测类私有字段和方法的能力(需配合注解使用,见下文详述)。
|
|
|
|
|
|
|
|
|
|
若要开启极速Mock功能,还需在`pom.xml`里加上`testable-maven-plugin`插件。
|
|
|
|
|
|
|
|
|
|
```xml
|
|
|
|
|
<plugin>
|
|
|
|
|
<groupId>com.alibaba.testable</groupId>
|
|
|
|
|
<artifactId>testable-maven-plugin</artifactId>
|
|
|
|
|
<version>${testable.version}</version>
|
|
|
|
|
<executions>
|
|
|
|
|
<execution>
|
|
|
|
|
<id>prepare</id>
|
|
|
|
|
<goals>
|
|
|
|
|
<goal>prepare</goal>
|
|
|
|
|
</goals>
|
|
|
|
|
</execution>
|
|
|
|
|
</executions>
|
|
|
|
|
</plugin>
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## 使用Testable
|
|
|
|
|
|
|
|
|
|
`Testable`目前能为测试类提供两项增强能力:__直接访问被测类的私有成员__ 和 __极速Mock被测方法中的调用__
|
|
|
|
|
|
|
|
|
|
### 访问私有成员字段和方法
|
|
|
|
|
|
2020-10-25 14:24:21 +08:00
|
|
|
|
只需为测试类添加`@EnablePrivateAccess`注解,即可在测试用例中获得以下增强能力:
|
|
|
|
|
|
|
|
|
|
- 调用被测类的私有方法
|
|
|
|
|
- 读取被测类的私有成员
|
|
|
|
|
- 修改被测类的私有成员
|
|
|
|
|
- 修改被测类的常量成员(使用final或static final修饰的成员)
|
|
|
|
|
|
|
|
|
|
访问和修改私有、常量成员时,IDE可能会提示语法有误,但编译器将能够正常运行测试。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
|
|
|
|
若不希望看到IDE的语法错误提醒,或是在基于JVM的非Java语言项目里(譬如Kotlin语言),也可以借助`PrivateAccessor`工具类来实现私有成员的访问。
|
|
|
|
|
|
2020-10-27 06:54:36 +08:00
|
|
|
|
效果见示例项目文件`DemoServiceTest.java`中的`should_able_to_mock_private_method()`和`should_able_to_mock_private_field()`测试用例。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
|
|
|
|
### Mock被测类的任意方法调用
|
|
|
|
|
|
2020-10-27 23:02:31 +08:00
|
|
|
|
**1. <u>覆写任意类的方法调用</u>**
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
在测试类里定义一个有`@TestableMock`注解的普通方法,使它与需覆写的方法名称、参数、返回值类型完全一致,然后在其参数列表首位再增加一个类型为该方法原本所属对象类型的参数。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
此时被测类中所有对该需覆写方法的调用,将在单元测试运行时,将自动被替换为对上述自定义Mock方法的调用。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
**注意**:当遇到有两个需覆写的方法重名时,可将需覆写的方法名写到`@TestableMock`注解的`targetMethod`参数里,此时Mock方法自身就可以随意命名了。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 06:54:36 +08:00
|
|
|
|
示例项目文件`DemoServiceTest.java`中的`should_able_to_mock_common_method()`用例详细展示了这种用法。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 23:02:31 +08:00
|
|
|
|
**2. <u>覆写被测类自身的成员方法</u>**
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
有时候,在对某些方法进行测试时,希望将被测类自身的另外一些成员方法Mock掉。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
操作方法与前一种情况相同,Mock方法的第一个参数类型需与被测类相同,即可实现对被测类自身(不论是公有或私有)成员方法的覆写。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 06:54:36 +08:00
|
|
|
|
详见示例项目文件`DemoServiceTest.java`中的`should_able_to_mock_member_method()`用例。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 23:02:31 +08:00
|
|
|
|
**3. <u>覆写任意类的静态方法</u>**
|
|
|
|
|
|
|
|
|
|
对于静态方法的Mock与普通方法相同。但需要注意的是,对于静态方法,传入Mock方法的第一个参数实际值始终是`null`。
|
|
|
|
|
|
|
|
|
|
详见示例项目文件`DemoServiceTest.java`中的`should_able_to_mock_static_method()`用例。
|
|
|
|
|
|
|
|
|
|
**4. <u>覆写任意类的new操作</u>**
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
在测试类里定义一个有`@TestableMock`注解的普通方法,将注解的`targetMethod`参数写为"<init>",然后使该方法与要被创建类型的构造函数参数、返回值类型完全一致,方法名称随意。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-25 20:40:44 +08:00
|
|
|
|
此时被测类中所有用`new`创建指定类的操作(并使用了与Mock方法参数一致的构造函数)将被替换为对该自定义方法的调用。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 06:54:36 +08:00
|
|
|
|
详见示例项目文件`DemoServiceTest.java`中的`should_able_to_mock_new_object()`用例。
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
2020-10-27 23:02:31 +08:00
|
|
|
|
**5. <u>识别当前测试用例和调用来源</u>**
|
2020-10-23 23:22:53 +08:00
|
|
|
|
|
|
|
|
|
在Mock方法中可以通过`TestableTool.TEST_CASE`和`TestableTool.SOURCE_METHOD`来识别**当前运行的测试用例名称**和**进入该Mock方法前的被测类方法名称**,从而区分处理不同的调用场景。
|
|
|
|
|
|
|
|
|
|
详见示例项目文件`DemoServiceTest.java`中的`should_able_to_get_source_method_name()`和`should_able_to_get_test_case_name()`用例。
|