0

0

如何在构造函数中正确测试异常抛出行为

心靈之曲

心靈之曲

发布时间:2026-01-05 18:36:10

|

575人浏览过

|

来源于php中文网

原创

如何在构造函数中正确测试异常抛出行为

本文讲解如何为在构造函数中隐式调用、且内部捕获而非抛出异常的方法(如 `retrievetags()`)编写有效单元测试,重点解决“期望异常未被捕获”问题,并提供可验证的替代方案。

在您提供的代码中,BudgetTags 构造函数会立即调用 retrieveTags(),而该方法虽可能触发 FileNotFoundException(它是 IOException 的子类),但并未向上抛出异常,而是通过 catch (IOException e) 捕获并仅执行 System.out.println(...)。因此,assertThrows(IOException.class, ...) 必然失败——因为 retrieveTags() 方法本身从不抛出异常,其签名也未声明 throws IOException。

这意味着:您当前的测试逻辑存在根本性误解——不是“异常没被抛出”,而是异常被静默吞掉了。JUnit 的 assertThrows 只能捕获显式抛出(且未被捕获)的异常,对已处理的异常无能为力。

✅ 正确的测试策略:验证异常处理行为,而非异常本身

由于异常被 catch 块消化,我们应转而验证该处理逻辑是否按预期执行。最直接、可靠的方式是 捕获并断言 System.out 的输出内容

AI Code Reviewer
AI Code Reviewer

AI自动审核代码

下载
@Test
@DisplayName("File name doesn't exist → logs fatal exception")
void testRetrieveTag3() {
    String invalidPath = "test\no_file.csv";

    // 重定向 System.out 到 ByteArrayOutputStream,以便捕获输出
    ByteArrayOutputStream outContent = new ByteArrayOutputStream();
    System.setOut(new PrintStream(outContent));

    try {
        // 构造 BudgetTags 实例(触发 retrieveTags → FileNotFoundException → catch → println)
        new BudgetTags(invalidPath);

        // 断言输出包含预期错误信息
        String output = outContent.toString().trim();
        assertTrue(output.contains("Fatal exception:"), 
                   "Expected 'Fatal exception:' in console output, but got: " + output);
        assertTrue(output.contains("No such file or directory") || 
                   output.contains("The system cannot find the file specified"),
                   "Output should indicate file not found: " + output);
    } finally {
        // 恢复原始 System.out,避免影响其他测试
        System.setOut(System.out);
    }
}
⚠️ 注意事项:不要在生产代码中依赖 System.out 进行错误报告。建议重构 retrieveTags():移除 catch 块,改为 throws IOException,让调用方(如构造函数)决定如何处理。这样测试可回归 assertThrows,更符合健壮设计原则。若必须保留日志,应使用 SLF4J/Log4j 等日志框架,并在测试中使用 LogCaptor(如 logcaptor)捕获日志,比操作 System.out 更安全、可维护。构造函数中执行 I/O 是反模式。理想做法是将 retrieveTags() 提取为独立可测试方法,并在构造函数中仅做参数校验;或采用工厂/构建器模式延迟加载。

✅ 进阶重构建议(推荐)

public class BudgetTags implements BudgetTagsList {
    private final Set<String> tags = new TreeSet<>();
    private final String tagFilePath;

    public BudgetTags(String tagFilePath) throws IOException {
        this.tagFilePath = Objects.requireNonNull(tagFilePath);
        this.retrieveTags(); // now declares throws IOException
    }

    public void retrieveTags() throws IOException { // removed try-catch, declares exception
        try (BufferedReader br = new BufferedReader(new FileReader(tagFilePath))) {
            String line;
            while ((line = br.readLine()) != null) {
                String[] row = line.split(",", 2);
                if (row.length > 0) tags.add(row[0].trim());
            }
        }
    }
}

对应测试即可简洁、语义清晰地验证异常:

@Test
@DisplayName("Constructor throws FileNotFoundException for missing file")
void testConstructorThrowsOnMissingFile() {
    String invalidPath = "test/no_file.csv";
    FileNotFoundException thrown = assertThrows(FileNotFoundException.class, 
        () -> new BudgetTags(invalidPath));
    assertTrue(thrown.getMessage().toLowerCase().contains("no file"));
}

这种设计不仅使测试更直观,也提升了 API 的可靠性与可组合性——调用者明确知晓潜在失败点,并能选择重试、降级或上报。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
软件测试常用工具
软件测试常用工具

软件测试常用工具有Selenium、JUnit、Appium、JMeter、LoadRunner、Postman、TestNG、LoadUI、SoapUI、Cucumber和Robot Framework等等。测试人员可以根据具体的测试需求和技术栈选择适合的工具,提高测试效率和准确性 。

459

2023.10.13

java测试工具有哪些
java测试工具有哪些

java测试工具有JUnit、TestNG、Mockito、Selenium、Apache JMeter和Cucumber。php还给大家带来了java有关的教程,欢迎大家前来学习阅读,希望对大家能有所帮助。

312

2023.10.23

Java 单元测试
Java 单元测试

本专题聚焦 Java 在软件测试与持续集成流程中的实战应用,系统讲解 JUnit 单元测试框架、Mock 数据、集成测试、代码覆盖率分析、Maven 测试配置、CI/CD 流水线搭建(Jenkins、GitHub Actions)等关键内容。通过实战案例(如企业级项目自动化测试、持续交付流程搭建),帮助学习者掌握 Java 项目质量保障与自动化交付的完整体系。

28

2025.10.24

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

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

808

2024.01.03

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

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

26

2025.12.06

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

3

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

21

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

108

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

51

2026.03.04

热门下载

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

精品课程

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

共32课时 | 5.9万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.9万人学习

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

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