0

0

如何测试内部捕获的异常

聖光之護

聖光之護

发布时间:2025-11-26 15:39:26

|

491人浏览过

|

来源于php中文网

原创

如何测试内部捕获的异常

本文深入探讨了在单元测试中如何处理和验证被内部捕获的异常。当方法内部抛出异常但随即被 `try-catch` 块捕获并处理(例如仅记录日志)时,传统的 `assertThrows` 机制将无法直接验证。文章分析了这种设计模式带来的测试挑战,并提供了两种主要解决方案:首先是推荐通过重构代码以提高可测试性,例如使用 `Optional` 或自定义结果对象来明确指示操作结果;其次是针对无法立即重构的现有代码,探讨了通过验证日志输出或利用 `fail()` 方法来间接测试异常处理逻辑的策略。

理解问题:内部捕获异常的测试挑战

软件开发中,我们经常会遇到方法内部抛出异常,但这些异常被随后的 try-catch 块捕获并处理的情况。例如,异常可能被记录下来,但不会重新抛出或以其他方式向上层调用者传递。这种设计模式给单元测试带来了挑战,因为标准的 assertThrows 断言机制只能检测到从被测试方法中 抛出 的异常,而无法感知在方法内部被捕获的异常。

考虑以下示例代码:

Class A (调用者)

public class A {
    private static Logger logger = LoggerFactory.getLogger("A");
    private B b;

    public A() {
        b = new B();
    }

    public void methodA() {
        b.methodB();
        logger.info("A");
    }
}

Class B (被调用者,内部抛出并捕获异常)

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class B {
    private static Logger logger = LoggerFactory.getLogger("B");

    public B() {
        // 通常logger在构造函数中初始化一次
    }

    public void methodB() {
        try {
            // 模拟一个内部异常
            throw new Exception("NULL");
        } catch(Exception e) {
            // 异常被捕获并记录,但未重新抛出
            logger.info("Exception thrown internally in B", e);
        }
    }
}

当尝试使用 assertThrows 来测试 Class B 内部抛出的异常时,会遇到问题:

错误的测试示例

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class ATest { // 假设这是测试 A 或 B 的类
    @Test
    public void testExceptionInB() {
        B b = new B();
        // 尝试断言 b.methodB() 抛出异常,但它内部已经捕获了
        assertThrows(Exception.class, () -> b.methodB());
    }
}

执行上述测试会得到如下错误:

Expected java.lang.Exception to be thrown, but nothing was thrown.
org.opentest4j.AssertionFailedError: Expected java.lang.Exception to be thrown, but nothing was thrown.

这正是因为 methodB 中的 catch 块已经处理了异常,使得 methodB 方法本身正常完成,没有向外抛出任何异常。

核心问题:异常处理设计缺陷

上述 Class B 的设计模式,即在内部捕获并“吞噬”异常而不向调用者提供任何明确反馈,通常被认为是一种不良实践。这种设计存在以下几个问题:

  1. 可测试性差:如上所示,难以直接验证内部异常的发生。
  2. 错误信息丢失:调用者无法得知操作是否失败以及失败的具体原因。
  3. 调试困难:当系统行为异常时,由于错误信息被内部消化,定位问题变得更加复杂。
  4. 违反“命令-查询分离”原则:一个方法在执行某个命令时,应该通过返回值或异常明确其结果。

理想情况下,异常应该在能够处理它们的层次被捕获。如果一个方法无法完全处理异常,它应该重新抛出(可能封装为更具体的业务异常),或者通过返回值明确指示失败状态。

推荐方案:重构以提高可测试性

解决内部捕获异常测试问题的最佳方法是重构代码,使其遵循更好的异常处理实践,从而自然地提高可测试性。

方案一:返回 Optional 或自定义结果对象

如果一个操作可能失败但又不想强制调用者处理异常(例如,操作失败不认为是“异常”情况,而是一种预期结果),可以使用 Optional 或自定义结果对象来明确表示操作的成功或失败,并携带相关信息。

企业网站通用源码1.0
企业网站通用源码1.0

企业网站通用源码是以aspcms作为核心进行开发的asp企业网站源码。企业网站通用源码是一套界面设计非常漂亮的企业网站源码,是2016年下半年的又一力作,适合大部分的企业在制作网站是参考或使用,源码亲测完整可用,没有任何功能限制,程序内核使用的是aspcms,如果有不懂的地方或者有不会用的地方可以搜索aspcms的相关技术问题来解决。网站UI虽然不是特别细腻,但是网站整体格调非常立体,尤其是通观全

下载

Class B 重构示例 (使用 Optional)

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Optional;

public class BRefactored {
    private static Logger logger = LoggerFactory.getLogger("BRefactored");

    public BRefactored() {
        // 构造函数逻辑
    }

    /**
     * 执行操作并返回一个Optional,指示操作是否成功。
     * 如果操作失败,Optional为空,并记录错误。
     * @return 如果操作成功,返回一个非空的Optional;否则返回Optional.empty()。
     */
    public Optional methodB() {
        try {
            // 模拟一个内部异常
            throw new Exception("Simulated internal error");
            // 假设这里是成功路径,返回一些数据
            // return Optional.of("Operation successful");
        } catch(Exception e) {
            logger.error("Exception thrown internally in BRefactored: {}", e.getMessage());
            // 明确表示操作失败
            return Optional.empty();
        }
    }
}

Class A 相应修改示例

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Optional;

public class AModified {
    private static Logger logger = LoggerFactory.getLogger("AModified");
    private BRefactored bRefactored;

    public AModified() {
        bRefactored = new BRefactored();
    }

    public void methodA() {
        Optional result = bRefactored.methodB();
        if (result.isPresent()) {
            logger.info("B operation successful: {}", result.get());
        } else {
            logger.warn("B operation failed, handling gracefully in A.");
        }
        logger.info("A");
    }
}

重构后 Class B 的测试示例 现在,可以直接测试 methodB 的返回值来判断内部操作是否成功或失败:

import org.junit.jupiter.api.Test;
import java.util.Optional;
import static org.junit.jupiter.api.Assertions.*;

public class BRefactoredTest {
    @Test
    void testMethodB_failureScenario() {
        BRefactored b = new BRefactored();
        Optional result = b.methodB();
        // 断言Optional为空,表示操作失败
        assertTrue(result.isEmpty(), "methodB should return empty Optional on failure");
    }

    // 如果有成功路径,可以这样测试
    // @Test
    // void testMethodB_successScenario() {
    //     BRefactored b = new BRefactored();
    //     Optional result = b.methodB();
    //     assertTrue(result.isPresent(), "methodB should return non-empty Optional on success");
    //     assertEquals("Operation successful", result.get());
    // }
}

方案二:重新抛出特定异常

如果内部异常代表了一个调用者应该知道并可能需要处理的错误情况,那么应该捕获原始异常并重新抛出封装后的、更具业务含义的异常。

Class B 重构示例 (重新抛出业务异常)

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

// 自定义业务异常
class BusinessOperationException extends RuntimeException {
    public BusinessOperationException(String message, Throwable cause) {
        super(message, cause);
    }
}

public class BRefactoredThrows {
    private static Logger logger = LoggerFactory.getLogger("BRefactoredThrows");

    public BRefactoredThrows() {
        // 构造函数逻辑
    }

    /**
     * 执行操作。如果内部发生错误,则抛出 BusinessOperationException。
     * @throws BusinessOperationException 如果内部操作失败。
     */
    public void methodB() {
        try {
            // 模拟一个内部异常
            throw new IllegalStateException("Critical internal state error");
        } catch(Exception e) {
            logger.error("Exception caught internally, rethrowing as BusinessOperationException: {}", e.getMessage());
            // 捕获原始异常,封装并重新抛出
            throw new BusinessOperationException("Failed to perform B operation", e);
        }
    }
}

重构后 Class B 的测试示例 现在可以使用 assertThrows 直接测试 methodB 抛出的业务异常:

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;

public class BRefactoredThrowsTest {
    @Test
    void testMethodB_throwsBusinessOperationException() {
        BRefactoredThrows b = new BRefactoredThrows();
        // 断言 methodB 抛出 BusinessOperationException
        BusinessOperationException thrown = assertThrows(
            BusinessOperationException.class,
            () -> b.methodB(),
            "methodB should throw BusinessOperationException"
        );
        // 进一步验证异常信息或原因
        assertTrue(thrown.getMessage().contains("Failed to perform B operation"));
        assertTrue(thrown.getCause() instanceof IllegalStateException);
    }
}

替代方案:测试现有“坏设计”代码 (非理想但有时必要)

在某些情况下,我们可能无法立即重构现有代码,但仍需要为其编写测试。对于内部捕获并记录日志的异常,可以采用以下非理想但有时实用的测试策略。

方案一:验证日志输出

如果异常被捕获后会记录日志,那么可以通过验证日志系统是否接收到预期的错误消息来间接确认异常的发生。这通常需要一些额外的设置来捕获和检查日志。

概念性实现思路:

  1. 使用测试日志 Appender/Listener:许多日志框架(如 Logback, Log4j2)允许在测试环境中配置一个特殊的 Appender,它可以捕获日志事件并将其存储在内存中,供测试断言使用。
  2. 模拟 Logger:使用 Mockito 等工具模拟 Logger 实例。如果 Logger 是通过依赖注入或可在测试中替换的方式获取的,可以模拟它来验证 error() 或 warn() 方法是否被调用,以及调用时传入的参数是否符合预期。

示例 (使用 Mockito 模拟 Logger) 假设 Class B 可以通过构造函数注入 Logger:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class BInjectableLogger {
    private final Logger logger;

    public BInjectableLogger(Logger logger) {
        this.logger = logger;
    }

    public void methodB() {
        try {
            throw new Exception("NULL");
        } catch(Exception e) {
            logger.info("Exception thrown internally in BInjectableLogger", e);
        }
    }
}

测试示例

import org.junit.jupiter.api.Test;
import org.mockito.ArgumentCaptor;
import org.slf4j.Logger;

import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.assertTrue;

public class BInjectableLoggerTest {
    @Test
    void testMethodB_logsException() {
        // 创建一个模拟的Logger
        Logger mockLogger = mock(Logger.class);
        BInjectableLogger b = new BInjectableLogger(mockLogger);

        b.methodB();

        // 验证logger.info方法是否被调用
        // 捕获传递给info方法的参数
        ArgumentCaptor messageCaptor = ArgumentCaptor.forClass(String.class);
        ArgumentCaptor throwableCaptor = ArgumentCaptor.forClass(Throwable.class);

        // 验证info方法至少被调用一次,并捕获参数
        verify(mockLogger, times(1)).info(messageCaptor.capture(), throwableCaptor.capture());

        // 断言日志消息和捕获的异常类型
        assertTrue(messageCaptor.getValue().contains("Exception thrown internally"));
        assertTrue(throwableCaptor.getValue() instanceof Exception);
        assertTrue(throwableCaptor.getValue().getMessage().contains("NULL"));
    }
}

注意事项:如果 Logger 是通过 LoggerFactory.getLogger() 静态获取的,直接模拟会比较困难,可能需要 PowerMock 等工具,或者使用日志框架提供的测试工具。

方案二:使用 fail() 确保异常被捕获(基于原答案)

这种方法的目标是确保异常 确实被捕获没有逃逸 到调用的上层。如果 methodB 的 try-catch 块未能捕获到异常(例如,异常类型不匹配或 try 块逻辑改变导致异常在 catch 块外部抛出),那么测试将失败。

测试示例

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.fail; // 导入 fail

public class ATestOriginalDesign {
    @Test
    void testMethodB_exceptionIsSwallowed() {
        B b = new B(); // 使用原始的Class B
        try {
            b.methodB();
            // 如果 methodB 内部的异常被成功捕获并处理,
            // 那么代码会执行到这里,表示异常没有逃逸。
            // 此时,我们不希望测试失败,所以这里不需要 fail()。
            // 如果测试的目的是确保它 *不会* 抛出异常到外部,则此路径表示成功。
            // 如果测试的目的是确保它 *内部* 抛出了异常,这种方法并不能直接验证。
        } catch (Exception e) {
            // 如果 methodB 内部的 catch 块没有捕获异常,
            // 异常就会逃逸到这里,导致测试失败。
            fail("Exception was not swallowed by B.methodB(): " + e.getMessage());
        }
    }

    // 另一种更符合原答案意图的场景:
    // 如果我们想测试一个方法,它 *应该* 捕获并处理某个异常,
    // 但万一它没处理,我们就让测试失败。
    @Test
    void testMethodB_ensuresNoUnhandledExceptionEscapes() {
        B b = new B(); // 原始的 Class B
        try {
            b.methodB();
            // 如果代码执行到这里,说明 methodB 成功处理了内部异常,没有向外抛出。
            // 对于一个“吞噬”异常的设计,这正是我们期望的行为,所以测试通过。
        } catch (Exception e) {
            // 如果 methodB 没有捕获异常,异常就会逃逸到这里。
            // 此时,我们认为这是一个失败情况,因为 methodB 的设计目标是捕获它。
            fail("Expected B.methodB() to swallow its internal exception, but it escaped: " + e.getMessage());
        }
    }
}

此方法的局限性:这种方法实际上是在测试“异常没有逃逸”,而不是直接验证“内部抛出了异常”。对于 Class B 这种 总是 捕获异常的设计,testMethodB_ensuresNoUnhandledExceptionEscapes 这样的测试会 总是通过,因为它只是验证了 methodB 的 catch 块功能正常。它无法直接

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

279

2023.10.25

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

389

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

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

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

465

2024.01.03

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

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

12

2025.12.06

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

6

2026.01.15

公务员递补名单公布时间 公务员递补要求
公务员递补名单公布时间 公务员递补要求

公务员递补名单公布时间不固定,通常在面试前,由招录单位(如国家知识产权局、海关等)发布,依据是原入围考生放弃资格,会按笔试成绩从高到低递补,递补考生需按公告要求限时确认并提交材料,及时参加面试/体检等后续环节。要求核心是按招录单位公告及时响应、提交材料(确认书、资格复审材料)并准时参加面试。

37

2026.01.15

公务员调剂条件 2026调剂公告时间
公务员调剂条件 2026调剂公告时间

(一)符合拟调剂职位所要求的资格条件。 (二)公共科目笔试成绩同时达到拟调剂职位和原报考职位的合格分数线,且考试类别相同。 拟调剂职位设置了专业科目笔试条件的,专业科目笔试成绩还须同时达到合格分数线,且考试类别相同。 (三)未进入原报考职位面试人员名单。

51

2026.01.15

热门下载

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

精品课程

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

共23课时 | 2.5万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.4万人学习

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

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