PowerMock单元测试

Stella981
• 阅读 526

      在Java程序的单元测试中常用的mock工具有Mockito和EasyMock。但是这两种mock工具都无法实现对静态、final、私有方法或类的mock。因此有了功能强大的PowerMock工具。PowerMock并不是一个独立、全新的工具而是在Mockito和EasyMock的基础上进行的扩展,它分别有针对Mockito级EasyMock的扩展实现。本文主要介绍PowerMock的常见用法:

1.pom依赖

<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
        <version>4.10<version>
    <scope>test</scope>
</dependency>        
<dependency>
        <groupId>org.powermock</groupId>
        <artifactId>powermock-module-junit4</artifactId>
        <version>1.6.3</version>
        <scope>test</scope>
</dependency>    
<dependency>    
        <groupId>org.powermock</groupId>    
        <artifactId>powermock-api-mockito</artifactId>    
        <version>1.6.3</version>    
        <scope>test</scope>    
</dependency>

2.基本用法

2.1 测试类加上必要的注解

@RunWith(PowerMockRunner.class)
@PrepareForTest({ExtensionLoader.class,Configs.class})  //静态类,测试类本身(需要mock构造方法时)
@PowerMockIgnore("javax.management.*")

2.2 mock基本对象

测试 convert.convertAndHanlder(method,ref,bead)方法

@Mock
Convert convert;


when(convert.convertAndHanlder(eq(method),eq(ref),eq(bead))).thenReturn(bead);

2.3 mock无返回值的方法

@Mock
ProviderComposeConfig providerComposeConfig;


doNothing().when(providerComposeConfig).putSync(eq(providerConfig));

 2.4 mock方法执行异常场景

@Mock
ProviderComposeConfig providerComposeConfig;

doThrow(new Exception("执行失败")).when(providerComposeConfig).putSync(eq(providerConfig));

2.5 mock 静态方法

 测试Configs.getConfig(ProviderComposeConfig.class)方法

前提:@PrepareForTest注解中添加该类:ProviderComposeConfig.class

@Mock
ProviderComposeConfig configs;


PowerMockito.mockStatic(Configs.class);
when(Configs.getConfig(eq(ProviderComposeConfig.class))).thenReturn(configs);

mock无返回值的static方法

// "xxxUtil" 是类名 // "xxxStaticMethod" 是 static 方法的方法名 // 这里假设 "xxxStaticMethod" 需要两个参数,一个是 int 型,一个是 String 型 PowerMockito.doNothing().when(xxxUtil.class, "xxxStaticMethod", 1,"mysql");

mock可变参数传递

public class Demo {
    public static void test(String... args) {
        System.out.println(args);
    }
}

 测试方法:

@RunWith(PowerMockRunner.class)
@PrepareForTest({Demo.class})
@PowerMockIgnore("javax.management.*")
public class DemoTest {
    
    @Test
    public void test() throws Exception {
        PowerMockito.mockStatic(Demo.class);
        PowerMockito.doNothing().when(Demo.class,"test",eq("liu1"),eq("liu2"));
        Demo.test("liu1","liu2");
    }
    
}

 2.6 mock 构造方法

前提:@PrepareForTest注解中添加该类:HookContext.class

@Mock
HookContext context; 
 

//无参构造
PowerMockito.whenNew(HookContext.class).withNoArguments().thenReturn(context);

//有参构造
PowerMockito.whenNew(HookContext.class).withArguments(anystring()).thenReturn(context);

 ③完成实例

被测试类:

public class HandlerHook {

    public final static String id = "handler";

    @Override
    public Bead doCall(HookContext context, Bead bead, AccessId id) throws Throwable {
        ProviderComposeConfig configs = Configs.getConfig(ProviderComposeConfig.class);
        ProviderConfig config = configs == null ? null : configs.get(id);

        if (config == null) {
            // 即没有该服务,1.本地调用 2.客户端信息有误,访问IP:PORT 却没有服务
            return null;
        }

        // 拿到转换器,调用
        Convert convert = ExtensionLoader.getExtensionLoader(Convert.class).getExtensionById(ConvertProcessor.id);
        return convert.convertAndHanlder(config.getMethod(), config.getRef(), bead);
    }

}

 测试类:

@RunWith(PowerMockRunner.class)
@PrepareForTest({ExtensionLoader.class,Configs.class})
@PowerMockIgnore("javax.management.*")
public class HandlerHookTest {

    @Mock
    HookContext hookContext;
    
    @Mock
    Bead bead;
    
    @Mock
    AccessId accessId;
    
    @Mock
    ExtensionLoader<Convert> extensionLoader;
    
    @Mock
    ProviderComposeConfig configs;
    
    @Mock
    ProviderConfig config;
    
    @Mock
    Method method ;
    
    @Mock
    Object ref;
    
    @Mock
    Convert convert;
    
    private HandlerHook handlerHook = new HandlerHook();
    
    @Test
    public void doCall() throws Throwable {
        PowerMockito.mockStatic(Configs.class);
        when(Configs.getConfig(eq(ProviderComposeConfig.class))).thenReturn(configs);
        when(configs.get(eq(accessId))).thenReturn(config);
        
        PowerMockito.mockStatic(ExtensionLoader.class);
        when(ExtensionLoader.getExtensionLoader(eq(Convert.class))).thenReturn(extensionLoader);
        when(extensionLoader.getExtensionById(eq("map"))).thenReturn(convert);
        when(config.getMethod()).thenReturn(method);
        when(config.getRef()).thenReturn(ref);
        
        when(convert.convertAndHanlder(eq(method),eq(ref),eq(bead))).thenReturn(bead);
        assertEquals(bead, handlerHook.doCall(hookContext, bead, accessId));
    }
    
}
点赞
收藏
评论区
推荐文章
刚刚好 刚刚好
2个月前
css问题
1、在IOS中图片不显示(给图片加了圆角或者img没有父级)<div<imgsrc""/</divdiv{width:20px;height:20px;borderradius:20px;overflow:h
blmius blmius
1年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
晴空闲云 晴空闲云
2个月前
css中box-sizing解放盒子实际宽高计算
我们知道传统的盒子模型,如果增加内边距padding和边框border,那么会撑大整个盒子,造成盒子的宽度不好计算,在实务中特别不方便。boxsizing可以设置盒模型的方式,可以很好的设置固定宽高的盒模型。盒子宽高计算假如我们设置如下盒子:宽度和高度均为200px,那么这会这个盒子实际的宽高就都是200px。但是当我们设置这个盒子的边框和内间距的时候,那
艾木酱 艾木酱
1个月前
快速入门|使用MemFire Cloud构建React Native应用程序
MemFireCloud是一款提供云数据库,用户可以创建云数据库,并对数据库进行管理,还可以对数据库进行备份操作。它还提供后端即服务,用户可以在1分钟内新建一个应用,使用自动生成的API和SDK,访问云数据库、对象存储、用户认证与授权等功能,可专
Wesley13 Wesley13
1年前
MySQL查询按照指定规则排序
1.按照指定(单个)字段排序selectfromtable_nameorderiddesc;2.按照指定(多个)字段排序selectfromtable_nameorderiddesc,statusdesc;3.按照指定字段和规则排序selec
Wesley13 Wesley13
1年前
Java日期时间API系列36
  十二时辰,古代劳动人民把一昼夜划分成十二个时段,每一个时段叫一个时辰。二十四小时和十二时辰对照表:时辰时间24时制子时深夜11:00凌晨01:0023:0001:00丑时上午01:00上午03:0001:0003:00寅时上午03:00上午0
Stella981 Stella981
1年前
Angular material mat
IconIconNamematiconcode_add\_comment_addcommenticon<maticonadd\_comment</maticon_attach\_file_attachfileicon<maticonattach\_file</maticon_attach\
Wesley13 Wesley13
1年前
PHP中的NOW()函数
是否有一个PHP函数以与MySQL函数NOW()相同的格式返回日期和时间?我知道如何使用date()做到这一点,但是我问是否有一个仅用于此的函数。例如,返回:2009120100:00:001楼使用此功能:functiongetDatetimeNow(){
Wesley13 Wesley13
1年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
helloworld_34035044 helloworld_34035044
5个月前
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
helloworld_28799839 helloworld_28799839
2个月前
常用知识整理
Javascript判断对象是否为空jsObject.keys(myObject).length0经常使用的三元运算我们经常遇到处理表格列状态字段如status的时候可以用到vue