Java单元测试(3)mock进阶-静态、final、私有⽅法mock
mock进阶
前⾔
上⼀章讲了Spring-boot的starter test使⽤mock的⽅式mockito。但是mockito由于实现⽅式的原因(动态代理)不能⽀持静态、final、私有⽅法的mock。其实还有⼀种叫native⽅法,只是⼀般⾃⼰写native⽅法的地⽅不多,可能Android系统在这⽅⾯使⽤较多,⽐如游戏。查询了⼀些资料与笔者的以往经历,主要使⽤的有powerMock与jMockit。
1. powerMock
1.1. powerMock官⽅⽂档
powerMock在以前使⽤较多,最近反⽽使⽤少了,根本原因是不⽀持Junit5。官⽅最新版只⽀持Junit4,见
描述的很清楚,⽀持junit4版本,或者testNG,笔者在maven仓库看到junit5的⽀持jar,但是没⼈使⽤,更恶⼼的是有个版本号居然不能显⽰,看起来不是官⽅推送的。
所以笔者使⽤testNG来测试
1.2. powerMock demo模拟
pom⽂件如下,笔者依赖testNG与AssertJ。
<dependency>
<groupId&stng</groupId>
<artifactId>testng</artifactId>
<version>7.1.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-api-mockito2</artifactId>
<version>2.0.7</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.powermock</groupId>
<artifactId>powermock-module-testng</artifactId>
<version>2.0.7</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId&kito</groupId>
<artifactId>mockito-core</artifactId>
<version>3.3.3</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.assertj</groupId>
<artifactId>assertj-core</artifactId>
<version>3.15.0</version>
<scope>test</scope>
</dependency>
随意写⼀个⽅法,包含静态、私有、final⽅法。如果是final类,⽅式同final⽅法。package com.feng.demo;
public class User {
private String name;
public String getResult(String test){
return test +"\tjunit";
}
private String getPrivateName(String test){
return"123";springboot aop
}
public final String getFinalName(String str){
return"1235"+ str;
}
public static String getStaticName(String string){
return"12356"+ string;
}
}
构建test⽅法,testNG需要继承PowerMockTestCase。
package com.feng.demo;
kito.Mock;
import org.kito.PowerMockito;
import lassloader.annotations.PrepareForTest;
import stng.PowerMockTestCase;
import flect.Whitebox;
stng.annotations.Test;
import flect.Method;
import static api.Assertions.assertThat;
import kito.ArgumentMatchers.anyString;
import kito.ArgumentMatchers.argThat;
@PrepareForTest(User.class)//final⽅法准备,final类同理
public class UserTest extends PowerMockTestCase {
@Mock
private User user;
@Test
public void testTestGetResult(){
//由于次案例使⽤mockito,为powermock扩展,普通mock等同于mockito
PowerMockito.Result(anyString())).thenReturn("powerMock");
String result = Result("tom");
assertThat(result).isEqualTo("powerMock");
}
@Test
public void testGetFinalName(){
//final⽅法mock;这⾥同时mock了参数,跟final⽆关
PowerMockito.FinalName(argThat(s ->true))).thenReturn("powerMock");        String result = FinalName("tom");
assertThat(result).isEqualTo("powerMock");
}
@Test
public void testGetStaticName(){
//静态⽅法mock,先要设置需要mock的静态类
PowerMockito.StaticName(argThat(s ->true))).thenReturn("powerMock");        String result = StaticName("tom");
assertThat(result).isEqualTo("powerMock");
}
@Test
public void testGetPrivateName()throws Exception {
//mock私有⽅法
PowerMockito.when(user,"getPrivateName",anyString()).thenReturn("powerMock");
//私有⽅法实现单元测试,本质是反射调⽤
Method method = hod(User.class,"getPrivateName", String.class);
Object result = method.invoke(user,"12");
assertThat(result).isEqualTo("powerMock");
Object say = Whitebox.invokeMethod(user,"getPrivateName","12");
assertThat(result).isEqualTo("powerMock");
}
}
2. JMockit
重点介绍jmockit,功能⼗分强悍,只是测试代码有点不美观。不知道Spring-Boot推荐mockito⽽不是j
mockit的原因是否是这个。jmockit查资料说⽀持mock私有⽅法,但笔者通过1.49版本测试是不⽀持的。
jmockit的本质Record-Replay-Verification
1. Record: 录制类/对象的⽅法调⽤,在mockito⾥即为打桩。
2. Replay: 重放⽅法,即调⽤⽅法。
3. Verification: 验证。⽐如验证某个⽅法有没有被调⽤,调⽤多少次。
jmockit更新不是很频繁,最近更新是2019-12的1.49版本,⽀持junit5,这是jmockit1版本,⼀直在更新。
⽽且作者有jmockit2项⽬不知为啥2017年后不维护了。
2.1. jmockit demo
pom依赖
<!-- Jmockit -->
<dependency>
<groupId>org.jmockit</groupId>
<artifactId>jmockit</artifactId>
<version>1.49</version>
<scope>test</scope>
</dependency>
<!-- junit5 -->
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter</artifactId>
<version>5.6.2</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.assertj</groupId>
<artifactId>assertj-core</artifactId>
<version>3.15.0</version>
<scope>test</scope>
</dependency>
继续使⽤上⽂的User作为需要测试的类。
这⾥要注意,仅仅依赖jar在maven的test是不管⽤的,笔者调试发现JMockit在执⾏单元测试时没有初始化,甚⾄笔者
@ExtendWith(JMockitExtension.class)注⼊类都直接报错了。
查询官⽅⽂档:
笔者加⼊插件⽴马正常了,笔者查询很多博客,都没有这个,但根据笔者实践与官⽅⽂档是需要的。
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
<configuration>
<argLine>
-javaagent:"${settings.localRepository}"/org/jmockit/jmockit/1.49/jmockit-1.49.jar
</argLine>
</configuration>
</plugin>
</plugins>
</build>
⽽且官⽅还给出了单元测试代码覆盖率输出的配置:
其实现在⽽⾔⽤处不⼤,⼀般使⽤jacoco也可以在sonarqube配置。
2.2. @Mocked
@Mocked修饰类或者接⼝,类或者接⼝被修饰后,是全局的,以后这个类或者对象的⽅法调⽤就会⾛mocked的实例,不会再执⾏原来的⽅法。@Mocked⾮常霸道,⾃⼰new⼀个对象也不会⽣效了,全部被Mocked的实例接管。
返回结果jmockit也会处理:
1. 原始类型(short,int,float,double,long)返回0
2. String返回null
3. 其它引⽤类型,返回这个引⽤类型的Mocked对象,即将返回的其他对象也Mocked了。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。