testable-mock/docs/zh-cn/doc/frequently-asked-questions.md

4.1 KiB
Raw Blame History

常见使用问题

1. 如何Mock被测类中通过@Autowired初始化的字段?

直接创建被测类对象,然后利用TestableMock访问私有成员的能力直接给这些字段赋值即可。

2. TestableMock是否能够与其他Mock工具一起使用

TestableMock可与其他基于动态代理机制的Mock工具安全的共同使用譬如MockitoEasyMockMockRunner等皆属此范畴。

对于会修改类加载器或被测类字节码的Mock工具譬如PowerMockJMockit,尚无案例证明会与TestableMock发生冲突,但从原理来说二者可能存在不兼容风险,请谨慎使用。

3. 父类变量指向子类对象时如何实现Mock方法

在代码中,经常会有使用接口变量或父类变量指向子类实例,调用父类或子类方法的情况。

这时候遵循一个原则Mock方法的首个参数类型始终与发起调用的变量类型一致

因此不论实际被调用方法来自父类还是子类也不论子类是否覆写该方法。若发起调用的变量为父类型或接口类型则Mock方法的首个参数类型都应该使用相应的父类或接口类型。

参见Java和Kotlin示例中DemoInheritTest测试类的用例。

4. 如何Mock对于泛型方法(模板方法)

与普通方法的Mock方法相同直接在Mock方法上使用相同的泛型参数即可。

参见Java和Kotlin示例中DemoTemplateTest测试类的用例。

由于JVM存在泛型擦除机制对于Java项目也可以直接使用Object类型替代泛型参数见Java版DemoTemplateTest测试类中被注释掉的"第二种写法"示例。

5. 在Kotlin项目对String类中的方法进行Mock不生效

Kotlin语言中的String类型实际上是kotlin.String,而非java.lang.String。但在构建生成自字节码的时候又会被替换为Java的java.lang.String因此无论将Mock目标写为kotlin.Stringjava.lang.String均无法正常匹配到原始的被调用方法。

实际场景中需要对String类中的方法进行Mock的场景很少TestableMock暂未对这种情况做特别处理。

6. 当被Mock的方法被其它测试类间接调用时依然有效吗?

同样有效Mock的作用范围是整个测试运行过程。

例如测试类AaaTest中Mock了Aaa类的某些私有方法、以及某些公有方法中的外部调用;在另一个测试类BbbTest中测试Bbb类时,某些方法间接用到了Aaa类被Mock过的方法或调用此时实际调用的同样会是AaaTest类中定义的Mock方法。

7. TestableMock能否用于Android项目的测试

结合Roboelectric测试框架可使用。

Android系统的DalvikART虚拟机采用了与标准JVM不同的字节码体系会影响TestableMock的正常工作。Roboelectric框架能在普通JVM虚拟机上运行Android单元测试其速度比通过Android虚拟机运行单元测试快非常多绝大多数Android App的单元测试都在使用Roboelectric框架。

8. 使用Mock时候遇到"Attempt to access none-static member in mock method"错误?

当前TestableMock的设计不允许在Mock方法中访问测试类的非static成员因为Mock方法自身会在运行期被动态修改为static类型。然而有些Java语句包括构造块譬如new ArrayList<String>() {{ append("data"); }})、匿名函数(譬如list.stream().map(i -> i.get)等等会在编译过程中生成额外的成员方法调用导致Mock方法执行报错。

最简单的解决办法是将Mock方法本身也声明为static类型(这样动态生成的调用也会是static的,避免了以上错误),例如原方法定义为:

@MockMethod
private int getXxx(Demo self) {}

将其修改为:

@MockMethod
private static int getXxx(Demo self) {}

在下一个大迭代版本(即0.5版本将会在保持当前Mock体验的前提下对Mock的实现机制进行修改不再需要修改Mock方法为静态方法从而彻底解决此类报错问题。