使用 PowerMockito 模拟 new A() 行为

使用 PowerMockito 模拟 new A() 行为

在 Java 测试中,当你想模拟 new A() 实例化的对象行为时,可以使用 PowerMockito。下面是一个完整的示例:

完整实现方案

复制代码
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.Mockito;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;

import static org.junit.Assert.assertEquals;

// 1. 使用PowerMockRunner
@RunWith(PowerMockRunner.class)
// 2. 准备要修改字节码的类
@PrepareForTest(YourClass.class) // 包含对new A()调用的类
public class YourClassTest {

    @Mock
    private A mockA; // Mock对象

    @Test
    public void testYourMethod() throws Exception {
        // 3. 设置mock对象的返回值
        Mockito.when(mockA.count()).thenReturn(15); // 模拟返回15
        
        // 4. 模拟new A()操作
        PowerMockito.whenNew(A.class).withNoArguments().thenReturn(mockA);
        
        // 5. 创建被测类实例
        YourClass yourClass = new YourClass();
        
        // 6. 执行测试
        int result = yourClass.yourMethod();
        
        // 7. 验证结果
        assertEquals(15, result);
        
        // 可选:验证方法是否被调用
        Mockito.verify(mockA).count();
    }
}

class A {
    public int count() {
        return 0; // 实际实现
    }
}

class YourClass {
    public int yourMethod() {
        A a = new A(); // 需要被模拟的实例化
        return a.count();
    }
}

Maven 依赖配置

复制代码
<dependencies>
    <!-- JUnit -->
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.13.2</version>
        <scope>test</scope>
    </dependency>
    
    <!-- Mockito Core -->
    <dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-core</artifactId>
        <version>5.3.1</version>
        <scope>test</scope>
    </dependency>
    
    <!-- PowerMock -->
    <dependency>
        <groupId>org.powermock</groupId>
        <artifactId>powermock-module-junit4</artifactId>
        <version>2.0.9</version>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>org.powermock</groupId>
        <artifactId>powermock-api-mockito2</artifactId>
        <version>2.0.9</version>
        <scope>test</scope>
    </dependency>
</dependencies>

关键步骤解析

  1. 测试类配置​:

    • @RunWith(PowerMockRunner.class):使用PowerMock运行器
    • @PrepareForTest(YourClass.class):指定需要修改字节码的类
  2. 模拟构造函数​:

    复制代码
    PowerMockito.whenNew(A.class)   // 拦截对A类的实例化
        .withNoArguments()           // 使用无参构造函数
        .thenReturn(mockA);          // 返回模拟对象
  3. 设置方法返回值​:

    复制代码
    Mockito.when(mockA.count())  // 当调用count()方法时
           .thenReturn(15);      // 返回指定值

实际效果说明

yourMethod() 中的 new A() 执行时:

  1. PowerMock 会拦截构造函数调用
  2. 返回预先创建的 mockA 实例
  3. 调用 count() 方法时返回预设值 (15)

这样既不需要修改被测试类的代码,也能完全控制新创建对象的行为。

注意事项

  1. 测试效率:PowerMock 修改字节码,会使测试执行变慢
  2. 兼容性:确保 PowerMock 版本与 Mockito/JUnit 版本兼容
  3. 替代方案:优先考虑重构代码使用依赖注入
  4. 范围限制 :只能模拟 new 表达式直接出现的类

对于简单场景,推荐优先重构代码使用依赖注入。但当处理遗留代码或第三方库时,PowerMock 提供的这种能力非常有用。

相关推荐
Raners_12 分钟前
【Java代码审计(2)】MyBatis XML 注入审计
xml·java·安全·网络安全·mybatis
BillKu20 分钟前
Java读取Excel日期内容
java·开发语言·excel
ai小鬼头24 分钟前
如何重装旁路由系统并优化AIStarter部署:一步步教程
java·css·github
笑衬人心。40 分钟前
Hashtable 与 HashMap 的区别笔记
java·数据结构·笔记
金心靖晨43 分钟前
消息中间件优化高手笔记
java·数据库·笔记
lemon_sjdk2 小时前
Java飞机大战小游戏(升级版)
java·前端·python
LUCIAZZZ2 小时前
高性能网络模式-Reactor和Preactor
java·服务器·开发语言·网络·操作系统·计算机系统
Dcs2 小时前
Java 开发者必读:近期框架更新汇总(Spring gRPC、Micronaut、Quarkus 等)
java
Pi_Qiu_3 小时前
Python初学者笔记第十三期 -- (常用内置函数)
java·笔记·python
hsx6663 小时前
Android 基础筑基(一)
java