intellij JUnit mockito
在intellij越来越普及的情况下,利用JUnit在intellij中进行测试就显得很基础了,但网上的资料总有误导的地方,这里记录一下。
总体而言,要开始单元测试,可以分为三步,添加相关的插件,添加相关的依赖,编写测试方法,下面依序说下。
一、添加相关的插件
在intellij中利用JUnit进行测试,需要三个插件,Junit,用来执行测试用例,JUnitGenerator V2.0,用来生成测试用例,Coverage,用来生成测试报告。
安装插件完毕,还需要对JUnit进行适当的设置:
Junit Generator设置
Setting --》 Other Setting--》 Junit Generator
更改输出路径,
Output Path:
${SOURCEPATH}/../../test/java/${PACKAGE}/${FILENAME}
更改默认单元测试框架,
Default Template:
Junit 4
更改JUnit4的默认模板,
Junit 4
test. $entry.packageName $entry.packageName
<pre>$date</pre> <pre>$today</pre>
二,添加相关的依赖
在maven项目中,添加如下的依赖:
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-all</artifactId>
<version>1.3</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
<version>2.28.2</version>
<scope>test</scope>
</dependency>
三、开始编写测试方法
1.步骤
1)查看项目的project structure,确保:
Source Folders: src\main\java
Test Source Folders: src\test\java
2)在被测试的类中,选择 generate -> JUnit Test -> JUnit 4
3)在生成的测试类中编写相关的测试代码
四、JUnit的注解
1.断言
编写单元测试方法的一个根本不同在于要用断言来表达测试是否通过。
@Test
public void testAdd() throws Exception {
int value = new Util().add(2,4);
Assert.assertEquals(6,value);
} org.junit.Assert
assertTrue(String message, boolean condition)
assertEquals(String message, Object expected,
Object actual)
assertArrayEquals(String message, Object[] expecteds, Object[] actuals)
assertNull(String message, Object object)
assertSame(String message, Object expected, Object actual)
assertThat(T actual, Matcher<? super T> matcher)
JUnit提供了各式各样的断言供使用,选择合适的即可。
2.@test的子属性
如果要测试异常或者时间,则可以利用@test注解的子属性。
excepted属性,异常测试
@Test (expected = Exception.class)
public void testDivideException() throws Exception {
new Junit_Test().divide(3,0);
fail("除数为零没有抛出异常");
} timeout属性,超时测试
@Test (timeout = 1000)
public void testDivideTimeout() throws Exception {
new Junit_Test_Demo().divide(6,3);
}
3.常用注解
@Test,标明是一个测试方法 @Before,在每个测试方法执行前都执行 @After,在每个测试方法执行完都执行 @BeforeClass,在测试类一开始就执行 @AfterClass,在测试类执行完毕执行 @Ignore,暂时忽略这个测试方法
4.高级注解
Rule:可以用来扩展JUnit的功能,改变测试方法的行为;
@ClassRule,类级别,执行测试类的时候只调用一次被注解的Rule
@Rule,方法级别,每个测试方法执行的时候都会调用被注解的Rule 内置的Rule
TemporaryFolder,创建临时目录或文件
ExternalResource,在测试之前创建资源,并在测试完成后销毁
TestName ,获取目前测试方法的名字
TestWatcher ,在每个触发点执行自定义的逻辑
Verifier ,在测试执行完成之后做一些校验,以验证测试结果是不是正确
ErrorCollector ,收集多个错误,并在测试执行完后一次过显示出来
@RunWith:默认BlockJunit4ClassRunner,可以指定特殊的Runner; Suit,一次执行多个类的测试用例 Parameterized,批量指定多个待测参数 Category,对测试类中的被测试方法进行分类执行 Theories,为待测方法提供一组参数的排列组合
五,Mock
说到单元测试,就不能不提Mock,常用的mock框架很多,比较多用的是mockito和powermock。
Mockito
Mock技术框架,能让我们隔离外部依赖以便对我们自己的业务逻辑代码进行单元测试,在编写单元测试时,不需要再进行繁琐的初始化工作,在需要调用某一个接口时,直接模拟一个假方法,并任意指定方法的返回值。 Mockito的工作原理是通过创建依赖对象的proxy,所有的调用先经过proxy对象,proxy对象拦截了所有的请求再根据预设的返回值进行处理。Mockito对于Java接口使用接口代理的方式来模拟,对于Java类使用继承的方式来模拟(也即会创建一个新的Class类)。
PowerMock
PowerMock则在Mockito原有的基础上做了扩展,通过修改类字节码并使用自定义ClassLoader加载运行的方式来实现mock静态方法、final方法、private方法、系统类的功能。 从两者的项目结构中就可以看出,PowerMock直接依赖于Mockito,所以如果项目中已经导入了PowerMock包就不需要再单独导入Mockito包,如果两者同时导入还要小心PowerMock和Mockito不同版本之间的兼容问题。
1.流程
Mockito Mock
mock(Class classToMock);
mock(Class classToMock, String name) Stub
when(mock.someMethod()).thenReturn(value)
when(mock.someMethod()).thenThrow(new RuntimeException)
when(mock.someMethod()).thenAnswer() exec
首先要利用mock来构造依赖,其次利用when语句来构造stub,然后就可以执行测试方法了。
其实还可以利用spy来构造依赖,但与mock构造有不同的地方:
mock
对于未指定处理规则的调用会按方法返回值类型返回该类型的默认值(如int、long则返回0,boolean则返回false,对象则返回null,void则什么都不做)
spy
未指定处理规则时则会直接调用真实方法
2.最佳实践
如何进行单元测试,每个公司都有不同的做法,一些最佳实践总结如下:
AIR原则
automatic,必须使用断言,禁止使用输出进行验证
independent,UT之间没有调用关系和前后关系
repeatable,不与外界环境耦合,可随时执行
粒度
只对单个的类进行测试,不检查跨类或跨系统的交互
只对公开的接口进行测试
维护
新增代码需要补充单元测试
变更代码需要修正单元测试
编写单元测试的BCDE原则
border,边界测试,特殊取值、特殊时间点、数据顺序等
correct,正确的输入,并得到预期的结果
design,与设计文档结合,编写单元测试
error,强制错误信息输入(非法数据,异常流程),并得到预期结果
覆盖率
语句覆盖率达到70%,分支覆盖率达到100%
3.利用Mock的一个例子
假如有person,dao和service三个类,其中dao是个接口,service依赖于dao,具体如下:
public class Person {
private final int id;
private final String name;
public Person(int id, String name) {
this.id = id;
this.name = name;
}
public int getId() {
return id;
}
public String getName() {
return name;
}
}
public interface IPersonDao {
Person getPerson(int id);
boolean update(Person person);
}
public class PersonService {
private final IPersonDao personDao; public PersonService(IPersonDao personDao) {
this.personDao = personDao;
} public boolean update(int id, String name) {
Person person = personDao.getPerson(id);
if (person == null) {
return false;
}
Person personUpdate = new Person(person.getId(), name);
return personDao.update(personUpdate);
}
}
利用mock来测试
import org.junit.Test;
import org.junit.Before;
import org.junit.After;
import static org.junit.Assert.*;
import static org.mockito.Mockito.*; import org.mockito.Mock;
import org.mockito.Mockito.*;
import org.mockito.MockitoAnnotations; import java.util.List; /**
* PersonService Tester.
*
* @author <Authors name>
* @since <pre>07/01/2019</pre>
* @version 1.0
*/
public class PersonServiceTest { private IPersonDao mockDao;
private PersonService personService; @Before
public void before() throws Exception {
mockDao = mock(IPersonDao.class);
when(mockDao.getPerson(1)).thenReturn(new Person(1,"mst"));
when(mockDao.update(isA(Person.class))).thenReturn(true); personService = new PersonService(mockDao);
} @After
public void after() throws Exception {
} /**
*
* Method: update(int id, String name)
*
*/
@Test
public void testUpdate() throws Exception {
boolean result = personService.update(1,"md"); assertTrue(result); //验证是否执行过一次getPerson(1)
verify(mockDao,times(1)).getPerson(eq(1));
//验证是否执行过一次update
verify(mockDao,times(1)).update(isA(Person.class));
} @Test
public void testUpdateNotFind() throws Exception {
boolean result = personService.update(2, "md"); assertFalse( result); //验证是否执行过一次getPerson(1)
verify(mockDao, times(1)).getPerson(eq(2));
//验证是否执行过一次update
verify(mockDao, never()).update(isA(Person.class));
} @Mock
List list; public PersonServiceTest(){
MockitoAnnotations.initMocks(this);
} @Test
public void testList(){
when(list.add(isA(Object.class))).thenReturn(true);
list.add(2);
list.add(5);
boolean re = list.add(1);
assertTrue(re);
} }
其中演示了两种mock对象的构造,一种是手动构造,如22和27行,一种是注解构造,如69和72行。
在before方法中进行了依赖的mock和stub操作,进而在testUpdate中利用相应的依赖和stub进行测试逻辑的执行。
intellij JUnit mockito的更多相关文章
- JUnit + Mockito 单元测试(二)
摘自: http://blog.csdn.net/zhangxin09/article/details/42422643 版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[-] 入门 ...
- JUnit + Mockito 单元测试(二)(good)
import org.junit.Test; import org.mockito.Matchers; import org.mockito.Mockito; import java.util.Lis ...
- 基于Springboot+Junit+Mockito做单元测试
前言 前面的两篇文章讨论过< 为什么要写单元测试,何时写,写多细 >和<单元测试规范>,这篇文章介绍如何使用Springboot+Junit+Mockito做单元测试,案例选取 ...
- JUnit + Mockito 单元测试
原 JUnit + Mockito 单元测试(二) 2015年01月05日 17:26:02 sp42a 阅读数:60755 版权声明:本文为博主原创文章,未经博主允许不得转载. https://bl ...
- Junit mockito 测试Controller层方法有Pageable异常
1.问题 在使用MockMVC+Mockito模拟Service层返回的时候,当我们在Controller层中参数方法调用有Pageable对象的时候,我们会发现,我们没办法生成一个Pageable的 ...
- Junit mockito解耦合测试
Mock测试是单元测试的重要方法之一. 1.相关网址 官网:http://mockito.org/ 项目源码:https://github.com/mockito/mockito api:http:/ ...
- JUnit+Mockito结合测试Spring MVC Controller
[本文出自天外归云的博客园] 概要简述 利用JUnit结合Mockito,再加上spingframework自带的一些方法,就可以组合起来对Spring MVC中的Controller层进行测试. 在 ...
- Spring Test, JUnit, Mockito, Hamcrest 集成 Web 测试
关于Spring 3.2 1. Spring 3.2 及以上版本自动开启检测URL后缀,设置Response content-type功能, 如果不手动关闭这个功能,当url后缀与accept头不一致 ...
- junit mockito
package com.zendaimoney.util; import static org.mockito.Mockito.*;import static org.junit.Assert.*;i ...
随机推荐
- Vue日历组件的功能
本来呢,开开心心的写完了这个功能,然后发现elemeng更新了,增加了日历组件这个功能 我的内心机器奔溃,但是,element的日历组件太简单了,我感觉还是手撸一个吧,毕竟也不是很难 实现了显示农历, ...
- rac启动维护笔记
Ohasd.bin将产生4个代理启动相关的资源 (1) oraagent:负责ora.asm.ora.evmd.ora.gipcd.ora.gpnpd.ora.mdnsd的启动和管理 (2) ...
- MySQL数据备份概述
MySQL备份类型 热备份.温备份.冷备份 (根据服务器状态) 热备份:读.写不受影响: 温备份:仅可以执行读操作: 冷备份:离线备份:读.写操作均中止: 物理备份与逻辑备份 (从对象来分) 物理备份 ...
- php获取ssl验证的https页面的源码
$response = "https://faculty.xidian.edu.cn/system/resource/tsites/tsitesencrypt.jsp?id=_tsites_ ...
- Linux权限设置基础
chmod linux系统中文件或目录有两个属性:访问权限和文件所有者,简称“权限”和“归属”. 访问权限包括:读取.写入.可执行. 归属包括:属主(拥有该文件的用户).属组(拥有该文件的用户组) ...
- PAT 乙级 1042.字符统计 C++/Java
题目来源 请编写程序,找出一段给定文字中出现最频繁的那个英文字母. 输入格式: 输入在一行中给出一个长度不超过 1000 的字符串.字符串由 ASCII 码表中任意可见字符及空格组成,至少包含 1 个 ...
- 00-赵志勇机器学习-Logistics_Regression-data.txt(转载)
4.45925637575900 8.22541838354701 0 0.0432761720122110 6.30740040001402 0 6.99716180262699 9.3133933 ...
- 搭建代理服务器时的笔记,request使用笔记
request 请求笔记: 1.opation中使用form字段传参 对应 content-type': 'application/x-www-form-urlencoded',如果想要content ...
- js闭包理解与使用场景
要理解闭包首先要知道什么是函数的作用域链 因为有函数的作用域链存在,所以函数无论在哪里调用,函数都可以使用函数外部作用域的变量. 当一个函数被调用时,会创建一个执行环境及相应的作用域链.然后使用arg ...
- [codevs1286]郁闷的出纳员
题目描述 Description OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复 ...