GVKun编程网logo

如何使用PowerMockito模拟构造函数(powermock 构造函数)

11

在本文中,我们将为您详细介绍如何使用PowerMockito模拟构造函数的相关知识,并且为您解答关于powermock构造函数的疑问,此外,我们还会提供一些关于java–可以在非最终具体类中的Powe

在本文中,我们将为您详细介绍如何使用PowerMockito模拟构造函数的相关知识,并且为您解答关于powermock 构造函数的疑问,此外,我们还会提供一些关于java – 可以在非最终具体类中的Powermockito模拟最终方法?、java – 如何使用Mockito传递构造函数参数、JavaPowerMockito模拟单个静态方法和返回对象、Junit,Mockito,PowerMockito 进行单元测试的有用信息。

本文目录一览:

如何使用PowerMockito模拟构造函数(powermock 构造函数)

如何使用PowerMockito模拟构造函数(powermock 构造函数)

我正在尝试第一次使用PowerMockito模拟类构造函数,但是它不起作用。我当前的代码是:

public class Bar {
    public String getText() {
        return "Fail";
    }
}

public class Foo {
    public String getValue(){
        Bar bar= new Bar();
        return bar.getText();
    }

}

@RunWith(PowerMockRunner.class)
@PrepareForTest(Bar.class)
public class FooTest {
    private Foo foo;
    @Mock
    private Bar mockBar;

    @Before
    public void setUp() throws Exception {
        MockitoAnnotations.initMocks(this);
        PowerMockito.whenNew(Bar.class).withNoArguments().thenReturn(mockBar);
        foo= new Foo();
    }

    @Test
    public void testGetValue() throws Exception {
        when(mockBar.getText()).thenReturn("Success");
        assertEquals("Success",foo.getValue());

    }
}

测试失败,因为返回的值为“ Fail”。我的问题在哪里?

java – 可以在非最终具体类中的Powermockito模拟最终方法?

java – 可以在非最终具体类中的Powermockito模拟最终方法?

假设我有一个非最终的具体类,最后一个方法如下.
public class ABC {
  public final String myMethod(){
      return "test test";
  }
}

使用powermockito在junit中调用myMethod()可以返回别的东西吗?谢谢

解决方法

这样做:
@RunWith(powermockrunner.class)
@PrepareForTest(ABC.class)
public class ABCTest {

    @Test
    public void finalCouldBeMock() {
        final ABC abc = powermockito.mock(ABC.class);
        powermockito.when(abc.myMethod()).thenReturn("toto");
        assertEquals("toto",abc.myMethod());
    }
}

java – 如何使用Mockito传递构造函数参数

java – 如何使用Mockito传递构造函数参数

我想使用Mockito框架模拟一个类,它有几个构造函数参数.

如何在不为私有成员变量生成setter的情况下传递这些构造函数参数?

谢谢

最佳答案
你说你想要模拟一些但不是所有的方法.我不确定你为什么要这样做 – 如果你的班级是一个合作者,那么嘲笑整个班级是有意义的.或者如果它是SUT,你可能根本不想嘲笑它.

您正在寻找的是spy,而不是模拟.如果你决定使用间谍,你将从一个真实的对象中创建它,它已经使用你需要构造的任何参数构建.

但是在你考虑使用间谍之前,我建议你更仔细地考虑一下你正在测试的是什么,以及为什么你认为你需要用模拟实现替换你的一些方法,而不是其他方法.

JavaPowerMockito模拟单个静态方法和返回对象

JavaPowerMockito模拟单个静态方法和返回对象

我想从包含2个静态方法m1和m2的类中模拟静态方法m1。我希望方法m1返回一个对象。

我尝试了以下

1)

PowerMockito.mockStatic(Static.class, new Answer<Long>() {         @Override         public Long answer(InvocationOnMock invocation) throws Throwable {            return 1000l;         }      });

这将同时调用m1和m2,它们具有不同的返回类型,因此会给出返回类型不匹配错误。

2)PowerMockito.when(Static.m1(param1, param2)).thenReturn(1000l); 但是,执行m1时不会调用此方法。

3)PowerMockito.mockPartial(Static.class, "m1"); 给出了我不能从http://code.google.com/p/powermock/wiki/MockitoUsage获得的,无法提供模拟部分的编译器错误。

答案1

小编典典

你想要做的是1的一部分和2的全部的组合。

你需要使用PowerMockito.mockStatic为类的所有静态方法启用静态模拟。这意味着可以使用when-thenReturn语法对它们进行存根。

但是,当你调用尚未在模拟实例上显式存根的方法时,你正在使用的2个参数的mathStatic重载为Mockito / PowerMock应该执行的操作提供了默认策略。

从javadoc:

创建具有指定策略的类模拟,以解决交互问题。这是一个非常高级的功能,通常你不需要它来编写不错的测试。但是,在使用旧系统时可能会有所帮助。这是默认答案,因此仅当你不存根方法调用时才会使用它。

该默认默认磕碰的策略是只返回NULL,0或假的对象,数量和布尔值的方法。通过使用2-arg重载,你说的是“不,不,不,默认情况下,使用此Answer子类的answer方法获取默认值。它返回Long,因此,如果你有静态方法返回的值与长期存在问题。

而是使用模拟静态的1-arg版本启用静态方法的存根,然后使用when-thenReturn指定对特定方法执行的操作。例如:

import static org.mockito.Mockito.*;import org.junit.Test;import org.junit.runner.RunWith;import org.mockito.invocation.InvocationOnMock;import org.mockito.stubbing.Answer;import org.powermock.api.mockito.PowerMockito;import org.powermock.core.classloader.annotations.PrepareForTest;import org.powermock.modules.junit4.PowerMockRunner;class ClassWithStatics {  public static String getString() {    return "String";  }  public static int getInt() {    return 1;  }}@RunWith(PowerMockRunner.class)@PrepareForTest(ClassWithStatics.class)public class StubJustOneStatic {  @Test  public void test() {    PowerMockito.mockStatic(ClassWithStatics.class);    when(ClassWithStatics.getString()).thenReturn("Hello!");    System.out.println("String: " + ClassWithStatics.getString());    System.out.println("Int: " + ClassWithStatics.getInt());  }}

字符串型静态方法被存根以返回“ Hello!”,而整数型静态方法使用缺省的存根,返回0。

Junit,Mockito,PowerMockito 进行单元测试

Junit,Mockito,PowerMockito 进行单元测试

文档入口:

https://github.com/powermock/powermock/wiki

私有方法mock文档:

https://github.com/powermock/powermock/wiki/MockPrivate

静态方法mock文档

https://github.com/powermock/powermock/wiki/MockStatic

案例参考:

https://ehlxr.me/2017/07/25/use-introduction-of-powermock/

 

概述:

    介绍基于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

                http://site.mockito.org/

            2)使用powerMock ,增强 静态方法,私有方法能力;

                https://github.com/powermock/powermock

二 Demo 本程序依赖

Xml代码 

<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)被测试类

Java代码 

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)测试类

Java代码 

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

Java代码 

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

Java代码 

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)测试类

Java代码 

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);  
    }  
}   

 

今天的关于如何使用PowerMockito模拟构造函数powermock 构造函数的分享已经结束,谢谢您的关注,如果想了解更多关于java – 可以在非最终具体类中的Powermockito模拟最终方法?、java – 如何使用Mockito传递构造函数参数、JavaPowerMockito模拟单个静态方法和返回对象、Junit,Mockito,PowerMockito 进行单元测试的相关知识,请在本站进行查询。

本文标签: