0

0

如何正确单元测试捕获异常语句中的异常适配器

花韻仙語

花韻仙語

发布时间:2025-11-29 15:28:12

|

988人浏览过

|

来源于php中文网

原创

如何正确单元测试捕获异常语句中的异常适配器

本文详细探讨了在java单元测试中,如何正确模拟异常适配器以覆盖try-catch块中的异常处理逻辑。核心在于理解mockito中thenreturn()和thenthrow()的区别,当被测试方法通过适配器返回一个异常对象并自行抛出时,应使用thenreturn()来模拟适配器的行为,而非让适配器直接thenthrow(),从而确保测试的准确性和覆盖率。

软件开发中,异常处理是确保系统健壮性的关键环节。对包含异常处理逻辑的代码进行单元测试,特别是当异常通过适配器进行转换时,需要精确地模拟外部依赖的行为。本文将深入探讨如何正确地单元测试Java中try-catch块内涉及异常适配器的代码,以确保异常路径得到充分覆盖。

理解被测方法中的异常处理逻辑

考虑以下Java方法,它从一个客户端获取信息,并在遇到特定客户端或内部服务器错误时,通过一个异常适配器将其转换为服务层异常并重新抛出:

public Method execute(@NonNull final String test) throws ServiceException {
    Object object; // 假设Method类型可以存储这个object
    try {
        object = javaClient.fetchInfo(test);
    } catch (ClientException | InternalServerError e) {
        // 关键点:serviceExceptionAdapter.apply(e) 返回一个ServiceException实例
        throw serviceExceptionAdapter.apply(e);
    }
    return object;
}

在这个execute方法中,javaClient.fetchInfo(test)可能会抛出ClientException或InternalServerError。当这些异常被捕获时,serviceExceptionAdapter.apply(e)会被调用。根据代码结构,apply方法显然是接收一个异常,并返回一个ServiceException的实例,然后execute方法再将这个返回的ServiceException实例抛出

分析初始测试方法及其问题

为了测试上述catch块中的异常路径,特别是InternalServerError的情况,我们可能会编写如下的单元测试:

class ProxyTest {

    private ExceptionAdapter serviceExceptionAdapter;
    private JavaClient mockJavaClient;
    private Proxy proxy; // 假设Proxy是包含execute方法的类

    @BeforeEach
    void setup() {
        this.serviceExceptionAdapter = mock(ExceptionAdapter.class);
        this.mockJavaClient = mock(JavaClient.class);
        proxy = new Proxy(mockJavaClient, serviceExceptionAdapter);
    }

    @Test
    void test_InternalServerError() {
        String testInput = "someTestValue"; // 定义一个测试输入
        // 模拟javaClient抛出InternalServerError
        when(mockJavaClient.fetchInfo(any())).thenThrow(InternalServerError.class);

        // 尝试模拟异常适配器抛出ServiceException
        when(serviceExceptionAdapter.apply(any())).thenThrow(ServiceException.class);

        // 断言execute方法会抛出ServiceException
        assertThrows(ServiceException.class, () -> proxy.execute(testInput));
        // 验证适配器被调用了一次
        verify(serviceExceptionAdapter, times(1)).apply(any());
    }
}

尽管上述测试断言了ServiceException的抛出并验证了适配器的调用,但它可能无法正确覆盖catch块中的所有逻辑,甚至可能导致测试失败或误导。核心问题出在这一行:

when(serviceExceptionAdapter.apply(any())).thenThrow(ServiceException.class);

这里我们试图让serviceExceptionAdapter.apply(any())方法直接抛出一个ServiceException。然而,回顾被测方法:

码上飞
码上飞

码上飞(CodeFlying) 是一款AI自动化开发平台,通过自然语言描述即可自动生成完整应用程序。

下载
throw serviceExceptionAdapter.apply(e);

execute方法期望serviceExceptionAdapter.apply(e)返回一个ServiceException实例,然后由execute方法自身通过throw关键字将其抛出。如果适配器方法被模拟为直接thenThrow(),那么execute方法在调用apply时就会捕获到这个由Mockito模拟的异常,而不是按照其内部逻辑接收一个返回的异常对象并抛出。这与实际生产代码的行为不符。

解决方案:使用 thenReturn() 模拟返回值

正确的做法是模拟serviceExceptionAdapter.apply(e)方法返回一个ServiceException实例,而不是让它直接抛出。execute方法会接收到这个返回的异常实例,并按照其设计将其抛出。

修改后的测试代码如下:

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.assertThrows;

// 假设这些是您的异常和适配器接口
interface JavaClient {
    Object fetchInfo(String test) throws ClientException, InternalServerError;
}

interface ExceptionAdapter {
    ServiceException apply(Exception e);
}

class ServiceException extends Exception {
    public ServiceException(String message) { super(message); }
    public ServiceException(String message, Throwable cause) { super(message, cause); }
}
class ClientException extends Exception {}
class InternalServerError extends Exception {}

// Proxy类,包含要测试的execute方法
class Proxy {
    private final JavaClient javaClient;
    private final ExceptionAdapter serviceExceptionAdapter;

    public Proxy(JavaClient javaClient, ExceptionAdapter serviceExceptionAdapter) {
        this.javaClient = javaClient;
        this.serviceExceptionAdapter = serviceExceptionAdapter;
    }

    public Object execute(@NonNull final String test) throws ServiceException {
        Object object;
        try {
            object = javaClient.fetchInfo(test);
        } catch (ClientException | InternalServerError e) {
            throw serviceExceptionAdapter.apply(e);
        }
        return object;
    }
}

class ProxyTest {

    private ExceptionAdapter serviceExceptionAdapter;
    private JavaClient mockJavaClient;
    private Proxy proxy;

    @BeforeEach
    void setup() {
        this.serviceExceptionAdapter = mock(ExceptionAdapter.class);
        this.mockJavaClient = mock(JavaClient.class);
        proxy = new Proxy(mockJavaClient, serviceExceptionAdapter);
    }

    @Test
    void test_InternalServerError_withCorrectAdapterMocking() {
        String testInput = "someTestValue";

        // 1. 模拟javaClient抛出InternalServerError,触发catch块
        when(mockJavaClient.fetchInfo(any())).thenThrow(InternalServerError.class);

        // 2. 模拟serviceExceptionAdapter.apply(any()) 返回一个ServiceException实例
        // 这一步至关重要,它模拟了适配器“创建”并“返回”一个异常对象
        when(serviceExceptionAdapter.apply(any()))
           .thenReturn(new ServiceException("Mocked Service Exception from Adapter"));

        // 3. 断言proxy.execute()会抛出ServiceException
        assertThrows(ServiceException.class, () -> proxy.execute(testInput));

        // 4. 验证serviceExceptionAdapter.apply()方法被调用了一次
        verify(serviceExceptionAdapter, times(1)).apply(any());
    }
}

通过将thenThrow(ServiceException.class)改为thenReturn(new ServiceException(...)),我们准确地模拟了serviceExceptionAdapter.apply(e)的预期行为——返回一个ServiceException实例。这样,execute方法就能接收到这个实例并按照其内部逻辑将其抛出,从而确保了对catch块的正确单元测试覆盖。

总结与注意事项

  • 区分 thenReturn() 和 thenThrow(): 这是Mocking中最基础也最容易混淆的概念。thenReturn()用于模拟方法返回一个值,而thenThrow()用于模拟方法在执行时抛出一个异常。在测试异常适配器时,务必根据适配器方法的实际签名(是返回一个异常对象还是直接抛出异常)来选择正确的模拟方式。
  • 理解被测代码的意图: 在编写测试之前,清晰地理解被测方法(execute)如何与依赖(serviceExceptionAdapter)交互至关重要。throw serviceExceptionAdapter.apply(e)明确表示apply方法是返回一个异常对象。
  • 构造具体的异常实例: 在thenReturn(new ServiceException(...))中,建议传入具体的异常消息或原因,这有助于在调试失败测试时提供更多上下文信息。
  • 测试所有异常路径: 除了InternalServerError,如果javaClient.fetchInfo()还能抛出ClientException,也应该编写类似的测试用例来覆盖该路径。

通过以上方法,我们可以确保单元测试准确地反映代码的实际行为,从而提高测试的质量和可靠性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

469

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

13

2025.12.06

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

1

2026.01.28

Java 消息队列与异步架构实战
Java 消息队列与异步架构实战

本专题系统讲解 Java 在消息队列与异步系统架构中的核心应用,涵盖消息队列基本原理、Kafka 与 RabbitMQ 的使用场景对比、生产者与消费者模型、消息可靠性与顺序性保障、重复消费与幂等处理,以及在高并发系统中的异步解耦设计。通过实战案例,帮助学习者掌握 使用 Java 构建高吞吐、高可靠异步消息系统的完整思路。

1

2026.01.28

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

23

2026.01.27

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

120

2026.01.26

edge浏览器怎样设置主页 edge浏览器自定义设置教程
edge浏览器怎样设置主页 edge浏览器自定义设置教程

在Edge浏览器中设置主页,请依次点击右上角“...”图标 > 设置 > 开始、主页和新建标签页。在“Microsoft Edge 启动时”选择“打开以下页面”,点击“添加新页面”并输入网址。若要使用主页按钮,需在“外观”设置中开启“显示主页按钮”并设定网址。

51

2026.01.26

苹果官方查询网站 苹果手机正品激活查询入口
苹果官方查询网站 苹果手机正品激活查询入口

苹果官方查询网站主要通过 checkcoverage.apple.com/cn/zh/ 进行,可用于查询序列号(SN)对应的保修状态、激活日期及技术支持服务。此外,查找丢失设备请使用 iCloud.com/find,购买信息与物流可访问 Apple (中国大陆) 订单状态页面。

192

2026.01.26

npd人格什么意思 npd人格有什么特征
npd人格什么意思 npd人格有什么特征

NPD(Narcissistic Personality Disorder)即自恋型人格障碍,是一种心理健康问题,特点是极度夸大自我重要性、需要过度赞美与关注,同时极度缺乏共情能力,背后常掩藏着低自尊和不安全感,影响人际关系、工作和生活,通常在青少年时期开始显现,需由专业人士诊断。

7

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号