- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章SpringBoot 单元测试实战(Mockito,MockBean)由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
1)setup 。
2)执行操作 。
3)验证结果 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public
class
CalculatorTest {
Calculator mCalculator;
@Before
// setup
public
void
setup() {
mCalculator =
new
Calculator();
}
@Test
//assert 部分可以帮助我们验证一个结果
public
void
testAdd()
throws
Exception {
int
sum = mCalculator.add(
1
,
2
);
assertEquals(
3
, sum);
//为了简洁,往往会static import Assert里面的所有方法。
}
@Test
@Ignore
(
"not implemented yet"
)
// 测试时忽略该方法
public
void
testMultiply()
throws
Exception {
}
// 表示验证这个测试方法将抛出 IllegalArgumentException 异常,若没抛出,则测试失败
@Test
(expected = IllegalArgumentException.
class
)
public
void
test() {
mCalculator.divide(
4
,
0
);
}
}
|
@BeforeClass
在所有测试方法执行前执行一次,一般在其中写上整体初始化的代码。@AfterClass
在所有测试方法后执行一次,一般在其中写上销毁和释放资源的代码。
1
2
3
4
5
6
7
8
|
// 注意这两个都是静态方法
@BeforeClass
public
static
void
test(){
}
@AfterClass
public
static
void
test(){
}
|
@Before
在每个方法测试前执行,一般用来初始化方法(比如我们在测试别的方法时,类中与其他测试方法共享的值已经被改变,为了保证测试结果的有效性,我们会在@Before注解的方法中重置数据)@After
在每个测试方法执行后,在方法执行完成后要做的事情。@Test(timeout = 1000)
测试方法执行超过1000毫秒后算超时,测试将失败。@Test(expected = Exception.class)
测试方法期望得到的异常类,如果方法执行没有抛出指定的异常,则测试失败。@Ignore("not ready yet")
执行测试时将忽略掉此方法,如果用于修饰类,则忽略整个类。@Test
编写一般测试用例用。@RunWith
在 Junit 中有很多个 Runner,他们负责调用你的测试代码,每一个 Runner 都有各自的特殊功能,你根据需要选择不同的 Runner 来运行你的测试代码。如果我们只是简单的做普通 Java 测试,不涉及 Spring Web 项目,你可以省略 @RunWith 注解,你要根据需要选择不同的 Runner 来运行你的测试代码.
按照设计,Junit不指定test方法的执行顺序.
@FixMethodOrder(MethodSorters.JVM)
:保留测试方法的执行顺序为JVM返回的顺序。每次测试的执行顺序有可能会所不同。@FixMethodOrder(MethodSorters.NAME_ASCENDING)
:根据测试方法的方法名排序,按照词典排序规则(ASC,从小到大,递增)。Failure 是测试失败,Error 是程序出错.
Maven本身并不是一个单元测试框架,它只是在构建执行到特定生命周期阶段的时候,通过插件来执行JUnit或者TestNG的测试用例。这个插件就是maven-surefire-plugin,也可以称为测试运行器(Test Runner),它能兼容JUnit 3、JUnit 4以及TestNG.
在默认情况下,maven-surefire-plugin的test目标会自动执行测试源码路径(默认为src/test/java/)下所有符合一组命名模式的测试类。这组模式为:
首先我们项目一般都是 MVC 分层的,而单元测试主要是在 Dao 层和 Service 层上进行编写。从项目结构上来说,Service 层是依赖 Dao 层的,但是从单元测试角度,对某个 Service 进行单元的时候,他所有依赖的类都应该进行Mock。而 Dao 层单元测试就比较简单了,只依赖数据库中的数据.
Mockito是mocking框架,它让你用简洁的API做测试。而且Mockito简单易学,它可读性强和验证语法简洁.
Mockito 是一个针对 Java 的单元测试模拟框架,它与 EasyMock 和 jMock 很相似,都是为了简化单元测试过程中测试上下文 ( 或者称之为测试驱动函数以及桩函数 ) 的搭建而开发的工具 。
相对于 EasyMock 和 jMock,Mockito 的优点是通过在执行后校验哪些函数已经被调用,消除了对期望行为(expectations)的需要。其它的 mocking 库需要在执行前记录期望行为(expectations),而这导致了丑陋的初始化代码.
SpringBoot 中的 pom.xml 文件需要添加的依赖:
1
2
3
4
5
|
<
dependency
>
<
groupId
>org.springframework.boot</
groupId
>
<
artifactId
>spring-boot-starter-test</
artifactId
>
<
scope
>test</
scope
>
</
dependency
>
|
进入 spring-boot-starter-test-2.1.3.RELEASE.pom 可以看到该依赖中已经有单元测试所需的大部分依赖,如:
junit
mockito
hamcrest
若为其他 spring 项目,需要自己添加 Junit 和 mockito 项目.
方法名 | 描述 |
---|---|
Mockito.mock(classToMock) | 模拟对象 |
Mockito.verify(mock) | 验证行为是否发生 |
Mockito.when(methodCall).thenReturn(value1).thenReturn(value2) | 触发时第一次返回value1,第n次都返回value2 |
Mockito.doThrow(toBeThrown).when(mock).[method] | 模拟抛出异常。 |
Mockito.mock(classToMock,defaultAnswer) | 使用默认Answer模拟对象 |
Mockito.when(methodCall).thenReturn(value) | 参数匹配 |
Mockito.doReturn(toBeReturned).when(mock).[method] | 参数匹配(直接执行不判断) |
Mockito.when(methodCall).thenAnswer(answer)) | 预期回调接口生成期望值 |
Mockito.doAnswer(answer).when(methodCall).[method] | 预期回调接口生成期望值(直接执行不判断) |
Mockito.spy(Object) | 用spy监控真实对象,设置真实对象行为 |
Mockito.doNothing().when(mock).[method] | 不做任何返回 |
Mockito.doCallRealMethod().when(mock).[method] //等价于Mockito.when(mock.[method]).thenCallRealMethod(); | 调用真实的方法 |
reset(mock) | 重置mock |
验证行为是否发生 。
1
2
3
4
5
6
7
8
|
//模拟创建一个List对象
List<Integer> mock = Mockito.mock(List.
class
);
//调用mock对象的方法
mock.add(
1
);
mock.clear();
//验证方法是否执行
Mockito.verify(mock).add(
1
);
Mockito.verify(mock).clear();
|
多次触发返回不同值 。
1
2
3
4
5
6
7
8
|
//mock一个Iterator类
Iterator iterator = mock(Iterator.
class
);
//预设当iterator调用next()时第一次返回hello,第n次都返回world
Mockito.when(iterator.next()).thenReturn(
"hello"
).thenReturn(
"world"
);
//使用mock的对象
String result = iterator.next() +
" "
+ iterator.next() +
" "
+ iterator.next();
//验证结果
Assert.assertEquals(
"hello world world"
,result);
|
模拟抛出异常 。
1
2
3
4
5
6
7
|
@Test
(expected = IOException.
class
)
//期望报IO异常
public
void
when_thenThrow()
throws
IOException{
OutputStream mock = Mockito.mock(OutputStream.
class
);
//预设当流关闭时抛出异常
Mockito.doThrow(
new
IOException()).when(mock).close();
mock.close();
}
|
使用默认Answer模拟对象 。
RETURNS_DEEP_STUBS 是创建mock对象时的备选参数之一 。
以下方法deepstubsTest和deepstubsTest2是等价的 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
@Test
public
void
deepstubsTest(){
A a=Mockito.mock(A.
class
,Mockito.RETURNS_DEEP_STUBS);
Mockito.when(a.getB().getName()).thenReturn(
"Beijing"
);
Assert.assertEquals(
"Beijing"
,a.getB().getName());
}
@Test
public
void
deepstubsTest2(){
A a=Mockito.mock(A.
class
);
B b=Mockito.mock(B.
class
);
Mockito.when(a.getB()).thenReturn(b);
Mockito.when(b.getName()).thenReturn(
"Beijing"
);
Assert.assertEquals(
"Beijing"
,a.getB().getName());
}
class
A{
private
B b;
public
B getB(){
return
b;
}
public
void
setB(B b){
this
.b=b;
}
}
class
B{
private
String name;
public
String getName(){
return
name;
}
public
void
setName(String name){
this
.name = name;
}
public
String getSex(Integer sex){
if
(sex==
1
){
return
"man"
;
}
else
{
return
"woman"
;
}
}
}
|
参数匹配 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
@Test
public
void
with_arguments(){
B b = Mockito.mock(B.
class
);
//预设根据不同的参数返回不同的结果
Mockito.when(b.getSex(
1
)).thenReturn(
"男"
);
Mockito.when(b.getSex(
2
)).thenReturn(
"女"
);
Assert.assertEquals(
"男"
, b.getSex(
1
));
Assert.assertEquals(
"女"
, b.getSex(
2
));
//对于没有预设的情况会返回默认值
Assert.assertEquals(
null
, b.getSex(
0
));
}
class
B{
private
String name;
public
String getName(){
return
name;
}
public
void
setName(String name){
this
.name = name;
}
public
String getSex(Integer sex){
if
(sex==
1
){
return
"man"
;
}
else
{
return
"woman"
;
}
}
}
|
匹配任意参数 。
Mockito.anyInt()
任何 int 值 ;Mockito.anyLong()
任何 long 值 ;Mockito.anyString()
任何 String 值 ;Mockito.any(XXX.class) 任何 XXX 类型的值 等等.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
@Test
public
void
with_unspecified_arguments(){
List list = Mockito.mock(List.
class
);
//匹配任意参数
Mockito.when(list.get(Mockito.anyInt())).thenReturn(
1
);
Mockito.when(list.contains(Mockito.argThat(
new
IsValid()))).thenReturn(
true
);
Assert.assertEquals(
1
,list.get(
1
));
Assert.assertEquals(
1
,list.get(
999
));
Assert.assertTrue(list.contains(
1
));
Assert.assertTrue(!list.contains(
3
));
}
class
IsValid
extends
ArgumentMatcher<List>{
@Override
public
boolean
matches(Object obj) {
return
obj.equals(
1
) || obj.equals(
2
);
}
}
|
注意:使用了参数匹配,那么所有的参数都必须通过matchers来匹配 。
Mockito继承Matchers,anyInt()等均为Matchers方法 。
当传入两个参数,其中一个参数采用任意参数时,指定参数需要matchers来对比 。
1
2
3
4
5
6
|
Comparator comparator = mock(Comparator.
class
);
comparator.compare(
"nihao"
,
"hello"
);
//如果你使用了参数匹配,那么所有的参数都必须通过matchers来匹配
Mockito.verify(comparator).compare(Mockito.anyString(),Mockito.eq(
"hello"
));
//下面的为无效的参数匹配使用
//verify(comparator).compare(anyString(),"hello");
|
自定义参数匹配 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
@Test
public
void
argumentMatchersTest(){
//创建mock对象
List<String> mock = mock(List.
class
);
//argThat(Matches<T> matcher)方法用来应用自定义的规则,可以传入任何实现Matcher接口的实现类。
Mockito.when(mock.addAll(Mockito.argThat(
new
IsListofTwoElements()))).thenReturn(
true
);
Assert.assertTrue(mock.addAll(Arrays.asList(
"one"
,
"two"
,
"three"
)));
}
class
IsListofTwoElements
extends
ArgumentMatcher<List>
{
public
boolean
matches(Object list)
{
return
((List)list).size()==
3
;
}
}
|
预期回调接口生成期望值 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
@Test
public
void
answerTest(){
List mockList = Mockito.mock(List.
class
);
//使用方法预期回调接口生成期望值(Answer结构)
Mockito.when(mockList.get(Mockito.anyInt())).thenAnswer(
new
CustomAnswer());
Assert.assertEquals(
"hello world:0"
,mockList.get(
0
));
Assert.assertEquals(
"hello world:999"
,mockList.get(
999
));
}
private
class
CustomAnswer
implements
Answer<String> {
@Override
public
String answer(InvocationOnMock invocation)
throws
Throwable {
Object[] args = invocation.getArguments();
return
"hello world:"
+args[
0
];
}
}
|
等价于:(也可使用匿名内部类实现) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
@Test
public
void
answer_with_callback(){
//使用Answer来生成我们我们期望的返回
Mockito.when(mockList.get(Mockito.anyInt())).thenAnswer(
new
Answer<Object>() {
@Override
public
Object answer(InvocationOnMock invocation)
throws
Throwable {
Object[] args = invocation.getArguments();
return
"hello world:"
+args[
0
];
}
});
Assert.assertEquals(
"hello world:0"
,mockList.get(
0
));
Assert. assertEquals(
"hello world:999"
,mockList.get(
999
));
}
|
预期回调接口生成期望值(直接执行) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
@Test
public
void
testAnswer1(){
List<String> mock = Mockito.mock(List.
class
);
Mockito.doAnswer(
new
CustomAnswer()).when(mock).get(Mockito.anyInt());
Assert.assertEquals(
"大于三"
, mock.get(
4
));
Assert.assertEquals(
"小于三"
, mock.get(
2
));
}
public
class
CustomAnswer
implements
Answer<String> {
public
String answer(InvocationOnMock invocation)
throws
Throwable {
Object[] args = invocation.getArguments();
Integer num = (Integer)args[
0
];
if
( num>
3
){
return
"大于三"
;
}
else
{
return
"小于三"
;
}
}
}
|
修改对未预设的调用返回默认期望(指定返回值) 。
1
2
3
4
5
6
7
8
9
10
11
|
//mock对象使用Answer来对未预设的调用返回默认期望值
List mock = Mockito.mock(List.
class
,
new
Answer() {
@Override
public
Object answer(InvocationOnMock invocation)
throws
Throwable {
return
999
;
}
});
//下面的get(1)没有预设,通常情况下会返回NULL,但是使用了Answer改变了默认期望值
Assert.assertEquals(
999
, mock.get(
1
));
//下面的size()没有预设,通常情况下会返回0,但是使用了Answer改变了默认期望值
Assert.assertEquals(
999
,mock.size());
|
用spy监控真实对象,设置真实对象行为 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
@Test
(expected = IndexOutOfBoundsException.
class
)
public
void
spy_on_real_objects(){
List list =
new
LinkedList();
List spy = Mockito.spy(list);
//下面预设的spy.get(0)会报错,因为会调用真实对象的get(0),所以会抛出越界异常
//Mockito.when(spy.get(0)).thenReturn(3);
//使用doReturn-when可以避免when-thenReturn调用真实对象api
Mockito.doReturn(
999
).when(spy).get(
999
);
//预设size()期望值
Mockito.when(spy.size()).thenReturn(
100
);
//调用真实对象的api
spy.add(
1
);
spy.add(
2
);
Assert.assertEquals(
100
,spy.size());
Assert.assertEquals(
1
,spy.get(
0
));
Assert.assertEquals(
2
,spy.get(
1
));
Assert.assertEquals(
999
,spy.get(
999
));
}
|
不做任何返回 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
@Test
public
void
Test() {
A a = Mockito.mock(A.
class
);
//void 方法才能调用doNothing()
Mockito.doNothing().when(a).setName(Mockito.anyString());
a.setName(
"bb"
);
Assert.assertEquals(
"bb"
,a.getName());
}
class
A {
private
String name;
private
void
setName(String name){
this
.name = name;
}
private
String getName(){
return
name;
}
}
|
调用真实的方法 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
@Test
public
void
Test() {
A a = Mockito.mock(A.
class
);
//void 方法才能调用doNothing()
Mockito.when(a.getName()).thenReturn(
"bb"
);
Assert.assertEquals(
"bb"
,a.getName());
//等价于Mockito.when(a.getName()).thenCallRealMethod();
Mockito.doCallRealMethod().when(a).getName();
Assert.assertEquals(
"zhangsan"
,a.getName());
}
class
A {
public
String getName(){
return
"zhangsan"
;
}
}
|
重置 mock 。
1
2
3
4
5
6
7
8
9
10
|
@Test
public
void
reset_mock(){
List list = mock(List.
class
);
Mockito. when(list.size()).thenReturn(
10
);
list.add(
1
);
Assert.assertEquals(
10
,list.size());
//重置mock,清除所有的互动和预设
Mockito.reset(list);
Assert.assertEquals(
0
,list.size());
}
|
@Mock 注解 。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public
class
MockitoTest {
@Mock
private
List mockList;
//必须在基类中添加初始化mock的代码,否则报错mock的对象为NULL
public
MockitoTest(){
MockitoAnnotations.initMocks(
this
);
}
@Test
public
void
AnnoTest() {
mockList.add(
1
);
Mockito.verify(mockList).add(
1
);
}
}
|
指定测试类使用运行器:MockitoJUnitRunner 。
1
2
3
4
5
6
7
8
9
10
|
@RunWith
(MockitoJUnitRunner.
class
)
public
class
MockitoTest2 {
@Mock
private
List mockList;
@Test
public
void
shorthand(){
mockList.add(
1
);
Mockito.verify(mockList).add(
1
);
}
}
|
使用 @MockBean 可以解决单元测试中的一些依赖问题,示例如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
@RunWith
(SpringRunner.
class
)
@SpringBootTest
public
class
ServiceWithMockBeanTest {
@MockBean
SampleDependencyA dependencyA;
@Autowired
SampleService sampleService;
@Test
public
void
testDependency() {
when(dependencyA.getExternalValue(anyString())).thenReturn(
"mock val: A"
);
assertEquals(
"mock val: A"
, sampleService.foo());
}
}
|
@MockBean 只能 mock 本地的代码——或者说是自己写的代码,对于储存在库中而且又是以 Bean 的形式装配到代码中的类无能为力.
@SpyBean 解决了 SpringBoot 的单元测试中 @MockBean 不能 mock 库中自动装配的 Bean 的局限(目前还没需求,有需要的自己查阅资料).
以上为个人经验,希望能给大家一个参考,也希望大家多多支持我.
原文链接:https://www.jianshu.com/p/ecbd7b5a2021 。
最后此篇关于SpringBoot 单元测试实战(Mockito,MockBean)的文章就讲到这里了,如果你想了解更多关于SpringBoot 单元测试实战(Mockito,MockBean)的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我获得了一些源代码示例,我想测试一些功能。不幸的是,我在执行程序时遇到问题: 11:41:31 [linqus@ottsrvafq1 example]$ javac -g test/test.jav
我想测试ggplot生成的两个图是否相同。一种选择是在绘图对象上使用all.equal,但我宁愿进行更艰巨的测试以确保它们相同,这似乎是identical()为我提供的东西。 但是,当我测试使用相同d
我确实使用 JUnit5 执行我的 Maven 测试,其中所有测试类都有 @ExtendWith({ProcessExtension.class}) 注释。如果是这种情况,此扩展必须根据特殊逻辑使测试
在开始使用 Node.js 开发有用的东西之前,您的流程是什么?您是否在 VowJS、Expresso 上创建测试?你使用 Selenium 测试吗?什么时候? 我有兴趣获得一个很好的工作流程来开发我
这个问题已经有答案了: What is a NullPointerException, and how do I fix it? (12 个回答) 已关闭 3 年前。 基于示例here ,我尝试为我的
我正在考虑测试一些 Vue.js 组件,作为 Laravel 应用程序的一部分。所以,我有一个在 Blade 模板中使用并生成 GET 的组件。在 mounted 期间请求生命周期钩子(Hook)。假
考虑以下程序: #include struct Test { int a; }; int main() { Test t=Test(); std::cout<
我目前的立场是:如果我使用 web 测试(在我的例子中可能是通过 VS.NET'08 测试工具和 WatiN)以及代码覆盖率和广泛的数据来彻底测试我的 ASP.NET 应用程序,我应该不需要编写单独的
我正在使用 C#、.NET 4.7 我有 3 个字符串,即。 [test.1, test.10, test.2] 我需要对它们进行排序以获得: test.1 test.2 test.10 我可能会得到
我有一个 ID 为“rv_list”的 RecyclerView。单击任何 RecyclerView 项目时,每个项目内都有一个可见的 id 为“star”的 View 。 我想用 expresso
我正在使用 Jest 和模拟器测试 Firebase 函数,尽管这些测试可能来自竞争条件。所谓 flakey,我的意思是有时它们会通过,有时不会,即使在同一台机器上也是如此。 测试和函数是用 Type
我在测试我与 typeahead.js ( https://github.com/angular-ui/bootstrap/blob/master/src/typeahead/typeahead.js
我正在尝试使用 Teamcity 自动运行测试,但似乎当代理编译项目时,它没有正确完成,因为当我运行运行测试之类的命令时,我收到以下错误: fatal error: 'Pushwoosh/PushNo
这是我第一次玩 cucumber ,还创建了一个测试和 API 的套件。我的问题是在测试 API 时是否需要运行它? 例如我脑子里有这个, 启动 express 服务器作为后台任务 然后当它启动时(我
我有我的主要应用程序项目,然后是我的测试的第二个项目。将所有类型的测试存储在该测试项目中是一种好的做法,还是应该将一些测试驻留在主应用程序项目中? 我应该在我的主项目中保留 POJO JUnit(测试
我正在努力弄清楚如何实现这个计数。模型是用户、测试、等级 用户 has_many 测试,测试 has_many 成绩。 每个等级都有一个计算分数(strong_pass、pass、fail、stron
我正在尝试测试一些涉及 OkHttp3 的下载代码,但不幸失败了。目标:测试 下载图像文件并验证其是否有效。平台:安卓。此代码可在生产环境中运行,但测试代码没有任何意义。 产品代码 class Fil
当我想为 iOS 运行 UI 测试时,我收到以下消息: SetUp : System.Exception : Unable to determine simulator version for X 堆
我正在使用 Firebase Remote Config 在 iOS 上设置 A/B 测试。 一切都已设置完毕,我正在 iOS 应用程序中读取服务器端默认值。 但是在多个模拟器上尝试,它们都读取了默认
[已编辑]:我已经用 promise 方式更改了我的代码。 我正在写 React with this starter 由 facebook 创建,我是测试方面的新手。 现在我有一个关于图像的组件,它有
我是一名优秀的程序员,十分优秀!