编写不易,转载请注明( http://shihlei.iteye.com/blog/2383925)!
概述:
介绍基于Junit ,Mockito,PowerMockito 常用的测试方法,包括异常测试,私有方法测试,没有返回值的方法测试,基于mock或spy的测试;
其中包含常规方法mock,私有方法mock。
一 测试介绍
1)测试原则:
凡是需要验证的方法都可以写单元测试,证明预期行为(不区分 controller,service ,dao)
2)测试分类:
a)单元测试:测试方法的 “逻辑” 是否满足期望。
b)集成测试:集成各个依赖组件,测试整体流程是否能通。
3)关于依赖:
单元测试一般测试当前方法的逻辑,不测试被依赖的类方法或自己的私有方法。
4)Mock 和 Spy 生成类区别
Mock:生成的类,所有方法都不是真实的方法,而且返回值都是NULL; 通过when 指定行为。
Spy:生成的类,所有方法都是真实方法,返回值都是和真实方法一样的;通过when 来将某个真实的方法,替换为指定行为的执行。
5)参考
1)普通mockito mock
2)使用powerMock ,增强 静态方法,私有方法能力;
二 Demo 本程序依赖
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>x.test</groupId> <artifactId>test</artifactId> <version>1.0-SNAPSHOT</version> <packaging>jar</packaging> <name>test</name> <url>http://maven.apache.org</url> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <junit.version>4.12</junit.version> <mockito.version>2.8.47</mockito.version> <powermock.version>1.7.0</powermock.version> <assertj.version>3.8.0</assertj.version> </properties> <dependencies> <!-- test --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>${junit.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>org.mockito</groupId> <artifactId>mockito-core</artifactId> <version>${mockito.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>org.powermock</groupId> <artifactId>powermock-module-junit4</artifactId> <version>${powermock.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>org.powermock</groupId> <artifactId>powermock-api-mockito2</artifactId> <version>${powermock.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>org.assertj</groupId> <artifactId>assertj-core</artifactId> <version>${assertj.version}</version> <scope>test</scope> </dependency> <!-- utils --> <dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId> <version>20.0</version> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <configuration> <source>1.8</source> <target>1.8</target> </configuration> </plugin> </plugins> </build> </project>
三 简单的测试
测试抛异常,无返回值的方法测试,私有方法测试
1)被测试类
package x.test.simple; import com.google.common.base.Strings; /** * 测试demo class * <p> * Created by shilei on 2017/7/10. */ public class HelloWho { private String who; public HelloWho(String who) { //条件路径 if (Strings.isNullOrEmpty(who)) { throw new IllegalArgumentException("who can not be null!"); } //赋值路径 this.who = who; } public void sayHello() { System.out.println(getHelloMessage()); } private String getHelloMessage() { return "Hello " + who; } }
2)测试类
package x.test.simple; import org.assertj.core.api.Assertions; import org.junit.Test; import org.powermock.reflect.Whitebox; /** * 基础测试: * <p> * 1 测试抛异常 * 2 测试无返回值 * 3 测试私有方法 */ public class HelloWhoTest { @Test(expected = IllegalArgumentException.class) public void test_construction_empty_who_throw_exception() { new HelloWho(""); } @Test(expected = IllegalArgumentException.class) public void test_construction_null_who_throw_exception() { new HelloWho(null); } /** * 无返回值的方法测试,查看其修改的成员变量是否成功 */ @Test public void test_construction_args_set_who() throws Exception { //预期结果 String expectWho = "A"; //执行被测试方法 HelloWho helloWho = new HelloWho(expectWho); //白盒,获取修改的成员变量 String actualWho = (String) Whitebox.getField(helloWho.getClass(), "who").get(helloWho); //断言 Assertions.assertThat(actualWho).isEqualTo(expectWho); } /** * 测试私有方法 */ @Test public void test_getHelloMessage() throws Exception { //准备方法参数 String who = "A"; //准备预期结构 String expextMessage = "Hello " + who; //执行被测试方法 HelloWho helloWho = new HelloWho(who); String actualMessage = Whitebox.invokeMethod(helloWho, "getHelloMessage"); //断言 Assertions.assertThat(actualMessage).isEqualTo(expextMessage); } }
四 基于Mock的测试
依赖mock,私有方法mock,验证方法是否被执行
1)被依赖类Dao
package x.test.adv.dao.pojo; /** * 用户 * <p> * Created by shilei on 2017/7/10. */ public class User { } package x.test.adv.dao; import x.test.adv.dao.pojo.User; /** * 用户dao * Created by shilei on 2017/7/10. */ public interface UserDao { /** * 保存 * * @param user 用户 * @return true : 保存成功 ; false : 用户已存在 */ boolean save(User user); }
2)待测试类Service
package x.test.adv.service; import x.test.adv.dao.pojo.User; /** * 用户服务 * <p> * Created by shilei on 2017/7/10. */ public interface UserService { /** * 保存 , 如果保存成功,记录日志 * * @param user user */ void save(User user); } package x.test.adv.service.impl; import x.test.adv.dao.UserDao; import x.test.adv.dao.pojo.User; import x.test.adv.service.UserService; /** * 带依赖的测试 * <p> * Created by shilei on 2017/7/10. */ public class UserServiceImpl implements UserService { private UserDao userDao; @Override public void save(User user) { //保存 boolean isSave = userDao.save(user); //写大数据日志 if (isSave) { writeLog(user); } } private void writeLog(User user) { System.out.println(user); } }
3)测试类
package x.test.adv.service.impl; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.InjectMocks; import org.mockito.Mock; import org.mockito.Mockito; import org.mockito.MockitoAnnotations; import org.powermock.api.mockito.PowerMockito; import org.powermock.core.classloader.annotations.PrepareForTest; import org.powermock.modules.junit4.PowerMockRunner; import x.test.adv.dao.UserDao; import x.test.adv.dao.pojo.User; /** * 带依赖的测试: * 1 mock 依赖 * 2 mock 私有方法 */ @RunWith(PowerMockRunner.class) //mock 静态方法,私有方法,需要添加该注解,以便通知框架 @PrepareForTest(UserServiceImpl.class) public class UserServiceImplTest { //mock 依赖注入到该bean @InjectMocks private UserServiceImpl userServiceImpl; //mock 依赖 @Mock private UserDao userDao; //完成依赖注入 @Before public void setUp() { MockitoAnnotations.initMocks(this); } /** * 测试保存成功,写日志 */ @Test public void test_save_true_verify_writelog() throws Exception { // 侦查对象 UserServiceImpl spyUserService = PowerMockito.spy(userServiceImpl); //创建输入 User user = new User(); //打桩 // mock 依赖行为,等价于 BDDMockito.given(userDao.save(user)).willReturn(true); Mockito.when(userDao.save(user)).thenReturn(true); //mock 私有方法 PowerMockito.doNothing().when(spyUserService, "writeLog", user); //调用实际方法 spyUserService.save(user); //断言 writelog 被执行一次 PowerMockito.verifyPrivate(spyUserService, Mockito.times(1)).invoke("writeLog", user); } /** * 测试保存成功,写日志 */ @Test public void test_save_false_no_writelog() throws Exception{ // 侦查对象 UserServiceImpl spyUserService = PowerMockito.spy(userServiceImpl); //创建输入 User user = new User(); //打桩 // mock 依赖行为,等价于 BDDMockito.given(userDao.save(user)).willReturn(true); Mockito.when(userDao.save(user)).thenReturn(false); //mock 私有方法 PowerMockito.doNothing().when(spyUserService, "writeLog", user); //调用实际方法 spyUserService.save(user); //断言 writelog 被执行一次 PowerMockito.verifyPrivate(spyUserService, Mockito.times(0)).invoke("writeLog", user); } }
相关推荐
PowerMock和Mockito是两个非常流行的Java单元测试框架,结合JUnit,它们可以提供强大的测试能力。在本文中,我们将深入探讨"PowerMock+Mockito-junit测试套件1.6.3版本"中的关键概念、功能和使用方法。 **PowerMock...
在Java领域,我们有许多优秀的单元测试框架,如JUnit、NUnit、XUnit,以及专门用于模拟对象的Mockito和PowerMockito。这些工具极大地简化了测试编写,使开发者能够隔离依赖,专注于待测代码的行为验证。下面我们将...
PowerMock是一个强大的Java单元测试框架,它扩展了如Mockito这样的工具,允许开发者模拟静态方法、构造函数、final类和方法、私有方法以及删除静态初始化器。在Java开发中,有时我们需要对不可Mock的对象进行单元...
在进行单元测试时,有些类或方法可能难以隔离,PowerMock则提供了这样的能力,使得几乎所有的Java代码都可以被测试。 接着,EasyMock是一个轻量级的模拟框架,它允许开发者创建并控制对象的行为。通过EasyMock,你...
使用在springboot项目中使用powermock-mockito进行单元测试,演示mock static、private、whenNew、exception的测试。 注意 示例使用spring-boot 2.0.3.RELEASE 它间接引用 junit-4.12 mockito-core-2.15.0 但是官网...
在单元测试领域,PowerMock是一个强大的工具,它扩展了诸如Mockito等模拟框架的功能,允许开发者模拟静态方法、构造函数、final类和方法、私有方法以及删除静态初始化器等。这一工具对于测试那些难以或者无法正常...
总结一下,Easymock3.1和PowerMock1.4.10是Java单元测试的重要工具,它们与JUnit配合使用,可以帮助开发者对各种复杂情况进行精确的测试,提高代码质量。在实际项目中,正确地使用这些库可以大大提升测试的覆盖率和...
PowerMockito是一个强大的Java单元测试框架,它扩展了Mockito的功能,允许我们模拟静态方法、构造函数、final类和方法,以及私有方法。在Java开发中,有时我们需要测试那些依赖于不可控外部因素(如系统时间、静态...
通过深入学习和实践`junit_powermock-master`项目,开发者可以掌握如何在复杂情况下使用JUnit和PowerMock进行单元测试,提升代码的测试覆盖率和整体质量。同时,理解并熟练运用这些工具,也有助于遵循良好的编程习惯...
然而,有些复杂的代码结构,如静态方法、final类或方法、构造器私有化等,使得传统的单元测试框架如JUnit和Mockito无法进行有效测试。这时,PowerMock就显得尤为必要。PowerMock是一个扩展了其他模拟库(如EasyMock...
本文将详细介绍如何利用PowerMock来模拟静态方法和私有方法,以便进行有效的单元测试。 PowerMock是一个强大的Java单元测试框架,它可以扩展其他流行的测试框架,如JUnit和TestNG。它的核心功能之一就是能够模拟...
可以使用Mockito模拟SqlSession,PowerMockito模拟Executor,再配合JUnit进行测试。 - 集成测试:模拟真实环境,测试整个应用的数据库操作流程,包括事务处理、缓存等。可以使用如Spring Boot Test、MyBatis-Spring...
PowerMock 是一个强大的Java单元测试框架,它扩展了其它单元测试工具,如Mockito,使得开发者可以模拟静态方法、构造函数、final类和方法、删除静态初始化器等。这款库在进行复杂对象模拟时非常有用,尤其是在处理...
在Java测试领域,有时我们需要对不可mock对象,如静态方法或final类,进行模拟以便于测试。PowerMock就是这样一款强大的工具,它扩展了EasyMock的功能,允许开发者在单元测试中模拟静态方法、构造器、final类和方法...
JUnit和Mockito则为单元测试提供支持,确保代码质量。而AssertJ和PowerMockito进一步提升了测试的质量和覆盖率。SpringData则使得数据访问变得更加便捷,无论使用哪种数据库。Spring MVC处理HTTP请求,实现Web服务的...
结合PowerMock进行测试时,可以对final方法进行mock。 ### Mockconstructors #### 使用场景 在需要控制类实例化过程时,PowerMock可以模拟构造函数的行为。 #### 业务代码 在业务代码中,可能需要确保对象以...
6. **Mocking**:在Java中,Mockito或PowerMockito等库可以模拟依赖,使测试更加独立,特别是在测试孤立功能时。 7. **静态代码分析**:PMD、Checkstyle和SonarQube等工具可以检查代码风格和潜在问题,提高代码质量...
3. **模拟方法**:在测试方法内部,使用 `PowerMockito.mockStatic()` 或 `PowerMockito.when()` 来模拟静态方法,`Mockito.when()` 用于模拟非静态方法。 4. **验证行为**:使用 `Mockito.verify()` 来检查方法是否...
**PowerMock** 是一个强大的Java单元测试工具,它扩展了JUnit的功能,使得开发者可以对静态方法、构造函数、final类/方法、私有方法、删除的方法以及由CGLIB支持的方法进行模拟和重定义。这对于复杂系统中的单元测试...
这个项目名为"testable-active-android",旨在展示如何在使用ActiveAndroid的同时,确保代码是可测试的,特别是通过使用PowerMock工具进行单元测试。 ActiveAndroid的主要功能是将Java对象映射到数据库表,允许...